如何强制JRuby使用java Longs而不是Fixnums初始化java ArrayList



我的团队中有人正在编写一个红宝石/黄瓜测试,该测试调用了一个接受List<Long>参数的java api。 我试图帮助他,但我的jruby知识非常有限。 每当调用需要List<Long>的java方法时,java方法都会抛出某种类强制转换异常,因为ArrayList包含所有Ruby Fixnum对象。

下面的简化代码显示我总是得到一个 Fixnum 而不是一个 Java::JavaLang::Long

irb(main):017:0> java.util.ArrayList.new([12.to_java])[0].class
=> Fixnum

即使以下内容显示12.to_java产生长

irb(main):018:0> 12.class
=> Fixnum
irb(main):019:0> 12.to_java.class
=> Java::JavaLang::Long 

我也尝试不使用构造函数参数

irb(main):020:0> a = java.util.ArrayList.new
=> #<Java::JavaUtil::ArrayList:0x314e60d2>
irb(main):021:0> a.add(12.to_java)
=> true
irb(main):022:0> a[0].class
=> Fixnum

并且直接实例化了一个java.lang.Long

irb(main):023:0> a = java.util.ArrayList.new
=> #<Java::JavaUtil::ArrayList:0xfdcb343>
irb(main):024:0> a.add(java.lang.Long.new(12))
=> true
irb(main):025:0> a[0].class
=> Fixnum

和我的最后一次尝试

irb(main):026:0> b = 12.to_java
=> #<Java::JavaLang::Long:0x244ff48e>
irb(main):027:0> b.class
=> Java::JavaLang::Long
irb(main):028:0> a = java.util.ArrayList.new
=> #<Java::JavaUtil::ArrayList:0x6a36ebaa>
irb(main):029:0> a.add b
=> true
irb(main):030:0> a.get(0).class
=> Fixnum

实际上,这比您想象的要容易得多,如果您只是在:)周围传递[],则可以正常工作

而在Ruby端

,JRuby确保值类似于Ruby,但是当它到达Java端时 - 如果RubyArray实例传递给Java(实际上实现了java.util.List) - 它将在最好的情况下自动转换"toJava"。 对于 Ruby Fixnums,您将java.lang.Long诸如java.util.List#get这样的操作:

public class JavaAPI {
    public void testListArgument(final List<Long> list) {
        System.out.println("list = " + list + " list.class = " + list.getClass() + " list[0].class = " + list.get(0).getClass());
    }
}

编译此.java(在目标/类下)而不是执行以下操作jruby -S irb

jruby-1.7.16 :007 > $CLASSPATH << 'target/classes'
 => ["file:/opt/local/rvm/rubies/jruby-1.7.16/lib/ruby/shared/readline/jline-2.11.jar", "file:/opt/local/rvm/rubies/jruby-1.7.16/lib/ruby/shared/readline/readline.jar", "file:target/classes/"] 
jruby-1.7.16 :009 > Java::JavaAPI.new.testListArgument [1]
list = [1] list.class = class org.jruby.RubyArray list[0].class = class java.lang.Long
 => nil 

最新更新