`
yangzhihuan
  • 浏览: 168663 次
  • 性别: 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  
楼主强,赞一个。 关于整数传递,看着代码有些难受,有没有优雅一点的解决方法?

相关推荐

    基于springboot教育资源共享平台源码数据库文档.zip

    基于springboot教育资源共享平台源码数据库文档.zip

    视频笔记linux开发篇

    linux开发篇,配套视频:https://www.bilibili.com/list/474327672?sid=4493702&spm_id_from=333.999.0.0&desc=1

    readera-24-09-08plus2020.apk

    ReadEra 这个阅读应用能够打开下列任何格式的文档: EPUB, PDF, DOC, RTF, TXT, DJVU, FB2, MOBI, 和 CHM. 基本上来说,你可以用它阅读你的设备内存中的任何书籍或者文本文档。 这个应用与划分成章节的文档兼。,有一个书签功能,可以在你阅读的时候,自动保存你的进度。另外,它让你更改页面模式,从几种不同的主题中进行挑选(夜间,白天,棕黑色调,还有控制台)。

    STM32单片机控制舵机旋转

    软件环境:KEIL4 硬件环境:STM32单片机+舵机 控制原理:通过控制输出信号的占空比调节舵机旋转的角度

    基于springboot仓库管理系统源码数据库文档.zip

    基于springboot仓库管理系统源码数据库文档.zip

    酒店管理系统源码C++实现的毕业设计项目源码.zip

    酒店管理系统源码C++实现的毕业设计项目源码.zip,个人大四的毕业设计、经导师指导并认可通过的高分设计项目,评审分98.5分。主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。 酒店管理系统源码C++实现的毕业设计项目源码.zip,酒店管理系统源码C++实现的毕业设计项目源码.zip个人大四的毕业设计、经导师指导并认可通过的高分设计项目,评审分98.5分。主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。酒店管理系统源码C++实现的毕业设计项目源码.zip酒店管理系统源码C++实现的毕业设计项目源码.zip酒店管理系统源码C++实现的毕业设计项目源码.zip,个人大四的毕业设计、经导师指导并认可通过的高分设计项目,评审分98.5分。主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。酒店管理系统源码C++实现的毕业设计项目源码.zip,个人大四的毕业设计、经导师指导并认可通过的高分设计项目,评审分98.5分。主要针对计算机相关专业的正在做毕

    58商铺全新UI试客试用平台网站源码

    58商铺全新UI试客试用平台网站源码

    基于SpringBoot+Vue的轻量级定时任务管理系统.zip

    springboot vue3前后端分离 基于SpringBoot+Vue的轻量级定时任务管理系统.zip

    毕业设计&课设_微博情感分析,用 flask 构建 restful api,含相关算法及数据文件.zip

    该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过严格测试运行成功才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。

    4D毫米波雷达点云数据处理方法研究.caj

    4D毫米波雷达点云数据处理方法研究.caj

    S M 2 2 5 8 X T量产工具

    S M 2 2 5 8 X T 量产工具供大家下载使用

    基于springboot的文物管理系统源码数据库文档.zip

    基于springboot的文物管理系统源码数据库文档.zip

    基于springboot的电影院售票管理系统源码数据库文档.zip

    基于springboot的电影院售票管理系统源码数据库文档.zip

    Javaweb仓库管理系统项目源码.zip

    基于Java web 实现的仓库管理系统源码,适用于初学者了解Java web的开发过程以及仓库管理系统的实现。

    美容美发项目,使用django框架,前后端一体化项目

    美容美发项目,使用django框架,前后端一体化项目

    2023年中国在线票务行业市场规模约为24.99亿元,挖掘市场新机遇

    在线票务:2023年中国在线票务行业市场规模约为24.99亿元,挖掘市场蓝海新机遇 在数字浪潮的席卷下,传统的票务销售模式正经历着前所未有的变革。纸质门票逐渐淡出人们的视野,取而代之的是便捷、高效的数字和移动票务。这一转变不仅为消费者带来了前所未有的购票体验,更为在线票务平台开辟了广阔的发展空间和市场机遇。随着国民经济的持续增长和文体娱乐行业的蓬勃发展,中国在线票务行业正站在时代的风口浪尖,等待着每一位有志之士的加入。那么,这片蓝海市场究竟蕴藏着怎样的潜力?又该如何把握机遇,实现突破?让我们一同探索。 市场概况: 近年来,中国在线票务行业市场规模持续扩大,展现出强劲的增长势头。据QYResearch数据显示,2023年中国在线票务行业市场规模约为24.99亿元,尽管受到宏观经济的影响,市场规模增速放缓,但整体趋势依然向好。这一增长主要得益于国民人均收入的不断提高、电影及演出行业的快速发展以及政府政策的支持。例如,2023年财政部、国家电影局发布的《关于阶段性免征国家电影事业发展专项资金政策的公告》,为电影行业注入了强劲动力,进而推动了在线票务市场规模的扩大。 技术创新与趋势: 技术进步

    基于SpringBoot的养老院管理系统源码数据库文档.zip

    基于SpringBoot的养老院管理系统源码数据库文档.zip

    毕业设计&课设_含构建设置及相关操作,基于特定技术,具体功能未详细说明.zip

    该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过严格测试运行成功才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。

    Go语言入门指南:基础语法、并发编程详解

    内容概要:本文档是一份详细的Go语言教程,从基础概念介绍到高级主题均有覆盖。主要内容包括Go语言的基础语法、数据类型、控制结构、函数、结构体、接口和并发编程等方面。通过具体示例介绍了如何使用Go语言进行开发。 适合人群:初学者和有一定经验的程序员都可以从这篇教程中受益,特别是那些想要快速掌握Go语言并应用于实际项目的开发者。 使用场景及目标:适用于初学者系统学习Go语言的基础知识和常用功能;也可以作为已有开发经验者的参考资料,帮助他们解决具体的编程问题,提高开发效率。 其他说明:本教程不仅包含了Go语言的基本知识点,还重点讲解了其独特的并发编程模型。读者在学习过程中应该注重理论与实践相结合,通过实际编写代码来加深理解和记忆。

    基于springboot计算机基础网上考试系统源码数据库文档.zip

    基于springboot计算机基础网上考试系统源码数据库文档.zip

Global site tag (gtag.js) - Google Analytics