`
yangzhihuan
  • 浏览: 169739 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

JRuby使用经验

阅读更多

首先我是一个Java程序员,很喜欢Ruby.

公司由于业务的需要,在Java项目中引入动态语言,目的是可以快速地修改业务逻辑以响应快速变化的业务需求.于是我有幸当了一回JRuby的先锋.当初使用JRuby的时候,我对JRuby项目的了解其实就是知道它可以让Ruby运行在JVM上面,其余细节一概不知,都是在实际使用中一点点地摸索,一点点地积累回来.

 

在这一过程中,在 dennis_zane 同学身上,我学到了很多与Ruby相关或者不相关的东西,借机感谢一下.

JRuby的中文资料相当的稀少,在 Google上搜索,来来去去的就是介绍了下最基本的怎么从Java中调用Ruby代码,或者在Ruby中使用Java的类库.我从无数次遇到问题 => 解决问题的循环中也有那么一点点的使用心得,记录之,备忘.

    * JRuby的入门资料,请访问 JRuby wiki 一般的使用方法这里都有介绍.
    * 有两种方法可以使用JRuby,一是用BSF,二是使用JDK 6.BSF的方式已经过时了,JDK6中内置了对脚本语言的支持,默认的Javascript,要使用JRuby还要下载juby-engine.jar,当前最新版本是1.1.6 地址: https://scripting.dev.java.net/files/documents/4957/115972/jruby-engine-1.1.6.zip

=======================================华丽的分割线========================================
如果使用jar打包,在ruby代码中调用java的类

 

        require "your_jar_file_name.jar"
        import your_packet_name
    



java 方法:

 

        class JavaClazz {
            public void javaMethod(int i) {
                System.out.pintln(i);
            }
        }
    


在Ruby中如是调用:

 

        java_clazz = JavaClazz.new
        java_clazz.javaMethod(1)
    


将会抛出类型不匹配的异常,因为所有ruby中的数值,传递到java那里都是 Long 类型,解决办法如下:

 

        java_clazz = JavaClazz.new
        java_clazz.javaMethod(java.lang.Integer.new(1))
    

注:以上代码是运行在 JRuby 1.1.2 版本下,在最新版本 1.2.0中已经没有这个问题了, 多谢 RednaxelaFX 同学的指正.
=======================================华丽的分割线========================================
如果在java中使用了可变参数:

 

        class JavaClazz {
            public void javaMethod(int i,String... s) {
                ... // your code
            }
        }
    


在ruby中应该这样调用:

 

        java_clazz = JavaClazz.new
        java_clazz.javaMethod(java.lang.Integer.new(1),'this is a string')
        // 只有一个参数,如果你知道java中的可变参数其实是一个数组的话
         java_clazz.javaMethod(java.lang.Integer.new(1),[].to_java(java.lang.String))
    

 

=======================================华丽的分割线========================================

调用java中的常量,枚举enum

 

        class JavaClazz {
            public final String CONSTANT = "I can not change!"
            public enum Season { winter, spring, summer, fall }
        }
    

 

 

        puts JavaClazz::CONSTANT
        puts JavaClazz::Season.winter
    


=======================================华丽的分割线========================================


如果你想使用ruby核心包,必须正确设置jruby的加载路径,从Sun实现的JRubyScriptEngine.java的源代码可以看到:

    //加载核心包的路径就是放在这个系统属性中的
    System.getProperty("com.sun.script.jruby.loadpath");
    //可以设置自己的路径
    System.setProperty("com.sun.script.jruby.loadpath","/root/.jruby/lib/ruby/1.8")


=======================================华丽的分割线========================================
关于官方JRuby引擎的问题
Sun官方实现的脚本引擎在多并发的情况下是会比较慢的,查看JRubyScriptEngine.java的源代码,可以看到eval方法是加上了synchronized

 

        public synchronized Object eval(Reader reader, ScriptContext ctx)
                       throws ScriptException {
            Node node = compileScript(reader, ctx);
            return evalNode(node, ctx);
        }
    


我至今想不明白,这个官方实现为什么会加上 synchronized

我自己山寨了一个JRubyScriptEngine的东西,直接调用JRuby的 JavaEmbedUtils 类来执行脚本,还是相当好用的.

/**
 *
 */
package org.opensource.script.jruby;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;

import org.jruby.Ruby;
import org.jruby.RubyRuntimeAdapter;
import org.jruby.javasupport.JavaEmbedUtils;
import org.jruby.runtime.GlobalVariable;
import org.jruby.runtime.builtin.IRubyObject;

/**
 * 山寨版JRubyScriptEngine
 * 弃用官方版的原因是由于它封装的invokeMethod方法使用了synchronized关键字,在多并发的情况下性能极差.
 *
 * @author yanghuan
 *
 */
public class JRubyScriptEngine {
    private final Ruby runtime;
    private final RubyRuntimeAdapter evaler;
    private final Map<String, IRubyObject> rubyObjectCache = new HashMap<String, IRubyObject>();

    public JRubyScriptEngine() {
        ArrayList<String> loadPaths = new ArrayList<String>();
        loadPaths.add("/root/.jruby/lib/ruby/1.8");
        loadPaths.add("/root/.jruby/lib/ruby/site_ruby/1.8");
        runtime = JavaEmbedUtils.initialize(loadPaths, JavaEmbedUtils
                .createClassCache(this.getClass().getClassLoader()));
        evaler = JavaEmbedUtils.newRuntimeAdapter();
    }

    /**
     * 根据脚本的路径,获取脚本eval后的Ruby对象,首先会从cache中检索,如有即时返回,如果没有则eval脚本,再返回.保证不会重复eval
     *
     * @param fullPath
     *            绝对路径
     * @return
     * @throws FileNotFoundException
     * @throws Exception
     */
    private IRubyObject getRubyObject(final String fullPath) throws FileNotFoundException {
        if (rubyObjectCache.get(fullPath) == null) {
            return evalScript(fullPath);
        }
        return rubyObjectCache.get(fullPath);
    }

    /**
     * 执行脚本,返回脚本对象 把这个方法从
     * #getRubyObject分出来,并且加上synchronized关键字,纯粹是为了防止多并发重复eval脚本
     *
     * @param fullPath
     * @return
     * @throws FileNotFoundException
     */
    private synchronized IRubyObject evalScript(final String fullPath) throws FileNotFoundException {
        if (rubyObjectCache.get(fullPath) == null) {
            File scriptFile = new File(fullPath);
            InputStream in = new FileInputStream(scriptFile);
            IRubyObject rubyObject = evaler.parse(runtime, in, scriptFile.getAbsolutePath(), 1).run();
            rubyObjectCache.put(fullPath, rubyObject);
            return rubyObject;
        }
        return rubyObjectCache.get(fullPath);
    }

    /**
     * 加载脚本
     *
     * @param fullPath
     * @throws FileNotFoundException
     */
    public void load(final String fullPath) throws FileNotFoundException {
        getRubyObject(fullPath);
    }

    /**
     * 清空已加载脚本对象
     *
     * @param fullPath
     */
    public void clean(final String fullPath) {
        if (rubyObjectCache.get(fullPath) != null) {
            rubyObjectCache.remove(fullPath);
        }
    }

    /**
     * 定义全局变量
     *
     * @param name
     *            变量名,不用以$开头
     * @param value
     *            值
     */
    public void defineGlobalVariable(final String name, final Object value) {
        IRubyObject rubyObject = JavaEmbedUtils.javaToRuby(runtime, value);
        /**
         * 这个全局变量的定义有点儿诡异,源代码是这样定义的:globalVariables.define(variable.name(),
         * newIAccessor() {}),所以必须手工加上 $ 开关
         **/
        GlobalVariable variable = new GlobalVariable(runtime, name.startsWith("$") ? name : "$" + name, rubyObject);
        runtime.defineVariable(variable);
    }

    /**
     * 执行脚本中定义的class的方法
     *
     * @param fullPath
     *            脚本绝对路径
     * @param method
     *            方法名
     * @param args
     *            参数
     * @return
     * @throws FileNotFoundException
     */
    public Object invokeMethod(final String fullPath, final String method, Object[] args) throws FileNotFoundException {
        IRubyObject rubyObject = getRubyObject(fullPath);
        return JavaEmbedUtils.invokeMethod(runtime, rubyObject, method, args, Object.class);
    }
}

 

 

分享到:
评论
5 楼 huaishk 2009-09-11  
一直不解Ruby,主要是没有机会使用。
期待将来在应用中使用Ruby。。。
4 楼 yangzhihuan 2009-05-15  
整型的转换有问题,是在 jruby 1.1.2 下面,已经注明了版本.

刚刚用jruby 1.2.0测试了一下已经没有问题了,应该是jruby已经修正了.
3 楼 RednaxelaFX 2009-05-15  
奇怪,之前我用JRuby的时候只是在可变长度参数那边痛苦过一阵,在整型上好像……我也不记得了。总之在Windows XP上用JRuby 1.1.5来测的话,
E:\sdk\jruby-1.1.5\bin>jirb
irb(main):001:0> JInteger = java.lang.Integer
=> Java::JavaLang::Integer
irb(main):002:0> JInteger.toString(1)
=> "1"
irb(main):003:0> JInteger.toString(18, 16)
=> "12"

java.lang.Integer的静态toString()就这俩重载,直接在JRuby 1.1.5里用也没问题。新的1.1.6、1.2和即将出来的1.3应该都还好吧?

用自己写的类也没问题:
public class Foo {
    public int increment(int i) {
        return i + 1;
    }
}

编译打包到Foo.jar后,
E:\sdk\jruby-1.1.5\bin>jirb
irb(main):001:0> require 'Foo.jar'
=> true
irb(main):002:0> Foo = Java::Foo
=> Java::Default::Foo
irb(main):003:0> f = Foo.new
=> #<Java::Default::Foo:0x3e926 @java_object=Foo@bfd66a>
irb(main):004:0> f.increment(1)
=> 2
irb(main):005:0> quit
2 楼 yangzhihuan 2009-05-15  
关于整数传递,还是有办法可以看代码好看些.
require "java"
include_class("java.lang.Integer") { |package, name| "J" + name }
i = JInteger.new(1)


不能使用
import java.lang.Integer
i = Integer.new(1)

是因为 Ruby 也有一个 Integer的类,会造成冲突,上面是使用重命名的方法来解决命名冲突的问题.

我比较懒,顺手就 java.lang.Integer 这样写了.
1 楼 popop 2009-05-14  
楼主强,赞一个。 关于整数传递,看着代码有些难受,有没有优雅一点的解决方法?

相关推荐

    jruby-src-1.0

    无论你是Ruby新手还是有经验的开发者,了解JRuby可以帮助你扩展技能树,将Ruby的优美特性带入到Java世界。 总的来说,"jruby-src-1.0.zip"是一个对于学习JRuby和Ruby在Java平台上的应用至关重要的资源。它不仅提供...

    jRuby On Rails WEB2.0

    通过JRuby on Rails,Java开发者可以使用Ruby的简洁语法和Rails的高效开发模式,同时享受Java的成熟技术栈和企业级支持。本书通过一系列实践项目,展示了如何将Rails应用与Java企业应用环境整合,包括如何利用Java ...

    JRuby 实战入门

    通过阅读《JRuby实战》这本书,你将获得丰富的实践经验和具体案例,帮助你快速掌握JRuby的使用,从而在实际项目中发挥其优势。这本书可能涵盖以上所有话题,并提供详细的示例代码和实践指导,确保你能够在短时间内从...

    走近JRuby

    【标题】:“走近JRuby” JRuby,全称为Java Ruby,是Ruby编程...通过这份PPT,读者不仅可以了解JRuby的基础知识,还能获取到实际操作中的宝贵经验和技巧,对想要深入了解和使用JRuby的开发者来说是一份宝贵的资源。

    JRuby on Rails Web 2.0 Projects

    JRuby使得开发人员能够在Java环境中使用Ruby语言编写应用程序,从而能够充分利用Ruby的灵活性以及Java平台的强大特性和广泛的支持。 #### JRuby on Rails的意义 Ruby on Rails(RoR)是一种流行的Web应用框架,以其...

    UsingJRubyFreePdfBook-英文原版.zip

    总的来说,《Using JRuby Free Pdf Book》是一本全面的JRuby指南,无论你是初学者还是经验丰富的开发者,都能从中获取到宝贵的知识和实践经验。通过阅读这本书,你可以掌握如何利用JRuby在Java平台上开发高效、可...

    PDF-JRubyCookbookFreePdfBook-英文版.rar

    《PDF-JRubyCookbookFreePdfBook-英文版》是一本专为JRuby开发者准备的实践指南,它以PDF格式提供,旨在帮助读者...无论你是初学者还是经验丰富的开发者,这本书都能提供宝贵的指导,帮助你提升JRuby开发的专业水平。

    word源码java-rubydoop:用JRuby编写Hadoop作业

    经验。 Rubydoop 的目标不是在 Hadoop 之上做一些新的事情,而是一种从 JRuby 使用 Hadoop 的方法。 如果您愿意,可以随意编写一些很棒的东西,使 Hadoop 更易于使用。 Rubydoop 并不完整。 配置 DSL 仅提供基本的...

    jading:cascading.jruby构建和执行工具

    原始代码和文档可能仍可通过链接获取,对于想要了解Cascading与JRuby结合使用或者想在其基础上进行二次开发的开发者,这些资料仍然是宝贵的。 **学习和使用建议** 1. **熟悉Cascading**:理解Cascading的基本概念...

    march_hare:适用于RabbitMQ的惯用语,快速且维护良好的JRuby客户端

    它致力于将Java客户端的强大功能与4年以上使用和开发和的经验相结合。 为什么选择三月野兔 JVM上的并发支持非常出色,它提供了许多工具和方法。 让我们利用它。 RabbitMQ Java客户端坚如磐石,并支持所有RabbitMQ...

    Rails相关电子书汇总二

    压缩包子文件的文件名称“Raven Scripting Java Builds with Ruby.pdf”揭示了一本书籍,该书可能探讨了如何使用Ruby(可能通过JRuby)进行Java构建。Raven可能是一个项目名称,Scripting可能是指使用脚本语言,而...

    Pro.Android.Python.with.SL4A(第1版)

    《Pro.Android.Python.with.SL4A(第1版)》是一本专注于使用Python语言以及少量JavaScript在Android平台上开发实际应用的书籍。...无论是初学者还是有经验的开发者,都能从中找到有价值的信息,拓展自己的技能树。

    新工具 (1).pdf

    Fabric Engine的优点是可以让脚本语言开发者无需具备多线程C++开发经验,只需在程序要求高性能的部分使用特殊的KL语言编写,即可大幅度提升运行效率。使用Fabric Engine可以媲美使用多线程的C++,且可以媲美使用...

    冇问题.zip

    由于“冇问题”是黑老大先前开发“秘密”和“糗事百科”网站时所使用的团队的代号,所以这次发布的代码使用了这个名字作为代号。 目标 我希望能在将来把“冇问题”发展成一个具有高可定制性的微型论坛系统。 但由于...

    Red5说明文档 flex视频

    文档中包含了使用JRuby编写Red5应用程序的示例,如`application.rb`和`demoservice.rb`,展示了如何将Ruby代码集成到Red5中,扩展其功能。关于RTMPT(RTMP Tunneling Protocol)协议的说明,Red5文档提供了详细的...

    JavaOne2009大会资料-Services SOA Platform and Middleware Services2

    "Rethinking the ESB: Lessons Learned from Challenging the Limitations and Pitfalls"深入讨论了ESB的局限性和挑战,以及如何从中学到的经验教训。随着微服务架构的兴起,对传统ESB的角色进行了重新评估,寻求更...

    jdk7安装程序1

    7. **动态语言支持**:引入了`invokedynamic`指令,为运行时绑定方法提供了更高效的方式,对动态语言如Groovy、JRuby等的支持更加友好。 二、JDK 7的安装步骤 1. **下载JDK**:访问Oracle官网,找到JDK 7的下载...

Global site tag (gtag.js) - Google Analytics