考虑以下Scala代码:
package scala_java
object MyScala {
def setFunc(func: Int => String) {
func(10)
}
}
现在使用Java,我希望将MyScala
用作:
package scala_java;
public class MyJava {
public static void main(String [] args) {
MyScala.setFunc(myFunc); // This line gives an error
}
public static String myFunc(int someInt) {
return String.valueOf(someInt);
}
}
但是,上述方法不起作用(正如预期的那样,因为Java不允许进行函数式编程)。在Java中传递函数最简单的解决方法是什么?我想要一个通用的解决方案,适用于具有任意数量参数的函数。
编辑:Java 8的语法是否比下面讨论的经典解决方案更好?答案 0 :(得分:65)
在scala.runtime
包中,有一些名为AbstractFunction1
的抽象类,以及其他arities的抽象类。要从Java中使用它们,您只需覆盖apply
,如下所示:
Function1<Integer, String> f = new AbstractFunction1<Integer, String>() {
public String apply(Integer someInt) {
return myFunc(someInt);
}
};
如果您使用的是Java 8并希望使用Java 8 lambda语法,请查看https://github.com/scala/scala-java8-compat。
答案 1 :(得分:27)
您必须在Java中手动实例化Function1
。类似的东西:
final Function1<Integer, String> f = new Function1<Integer, String>() {
public int $tag() {
return Function1$class.$tag(this);
}
public <A> Function1<A, String> compose(Function1<A, Integer> f) {
return Function1$class.compose(this, f);
}
public String apply(Integer someInt) {
return myFunc(someInt);
}
};
MyScala.setFunc(f);
这取自Daniel Spiewak’s “Interop Between Java and Scala” article。
答案 2 :(得分:7)
对我来说最简单的方法是定义一个 java 界面,如:
public interface JFunction<A,B> {
public B compute( A a );
}
然后修改您的scala代码,重载setFunc
以接受JFunction
对象,例如:
object MyScala {
// API for scala
def setFunc(func: Int => String) {
func(10)
}
// API for java
def setFunc(jFunc: JFunction[Int,String]) {
setFunc( (i:Int) => jFunc.compute(i) )
}
}
您自然会使用scala中的第一个定义,但仍然可以使用java中的第二个定义:
public class MyJava {
public static void main(String [] args) {
MyScala.setFunc(myFunc); // This line gives an error
}
public static final JFunction<Integer,String> myFunc =
new JFunction<Integer,String>() {
public String compute( Integer a ) {
return String.valueOf(a);
}
};
}
答案 3 :(得分:0)
这是我尝试一个解决方案,一个小型库:https://github.com/eirslett/sc8
将Java 8 lambda包装在F(...)中,然后将其转换为Scala函数。