gpt4 book ai didi

reflection - 有没有一种简单(惯用)的方法将 java.lang.reflect.Method 转换为 Scala 函数?

转载 作者:行者123 更新时间:2023-12-03 12:37:51 28 4
gpt4 key购买 nike

我可以通过反射检索方法,以某种方式将其与目标对象结合起来,并将其作为看起来像 Scala 中的函数的东西返回(即您可以使用括号调用它)吗?参数列表是可变的。它不一定是“一流”函数(我已经更新了问题),只是一个看起来语法的函数调用,例如f(参数)。

到目前为止,我的尝试看起来像这样(从技术上讲,这是伪代码,只是为了避免附加定义使帖子变得困惑):

class method_ref(o: AnyRef, m: java.lang.reflect.Method) {    def apply(args: Any*): some_return_type = {        var oa: Array[Object] = args.toArray.map { _.asInstanceOf[Object] }        println("calling: " + m.toString + " with: " + oa.length)        m.invoke(o, oa: _*) match  {            case x: some_return_type => x;            case u => throw new Exception("unknown result" + u);        }    }}

通过上述方法,我能够克服编译器错误,但现在我遇到了运行时异常:

Caused by: java.lang.IllegalArgumentException: argument type mismatch

示例用法类似于:

  var f = ... some expression returning method_ref ...;  ...  var y = f(x) // looks like a function, doesn't it?

更新

将 args:Any* 更改为 args:AnyRef* 实际上解决了我的运行时问题,因此上述方法(经过修复)可以很好地满足我想要完成的任务。我想我在这里遇到了一个更普遍的可变参数问题。

最佳答案

当然。这是我编写的一些代码,用于向函数添加接口(interface)。这并不完全是你想要的,但我认为它可以通过一些改变来适应。最困难的更改是invoke,您需要将调用的方法更改为通过反射获得的方法。另外,您还必须注意您正在处理的接收方法是apply。此外,您可以使用目标对象,而不是 f。它可能看起来像这样:

  def invoke(proxy: AnyRef, method: Method, args: Array[AnyRef]) = method match {
case m if /* m is apply */ => target.getClass().getMethod("name", /* parameter type */).invoke(target, args: _*)
case _ => /* ??? */
}

无论如何,这是代码:

import java.lang.reflect.{Proxy, InvocationHandler, Method}

class Handler[T, R](f: Function1[T, R])(implicit fm: Manifest[Function1[T, R]]) extends InvocationHandler {
def invoke(proxy: AnyRef, method: Method, args: Array[AnyRef]) = method.invoke(f, args: _*)
def withInterface[I](implicit m: Manifest[I]) = {
require(m <:< manifest[Function1[T, R]] && m.erasure.isInterface)
Proxy.newProxyInstance(m.erasure.getClassLoader(), Array(m.erasure), this).asInstanceOf[I]
}
}

object Handler {
def apply[T, R](f: Function1[T, R])(implicit fm: Manifest[Function1[T, R]]) = new Handler(f)
}

并像这样使用它:

trait CostFunction extends Function1[String, Int]
Handler { x: String => x.length } withInterface manifest[CostFunction]

那里使用“manifest”有助于语法。你可以这样写:

Handler({ x: String => x.length }).withInterface[CostFunction] // or
Handler((_: String).length).withInterface[CostFunction]

也可以删除 list 并使用 classOf 代替,并进行一些更改。

关于reflection - 有没有一种简单(惯用)的方法将 java.lang.reflect.Method 转换为 Scala 函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2661132/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com