如何在Any对象上使用applyDynamic

时间:2012-07-02 21:56:36

标签: scala dynamic

我正在考虑使用新的Type Dynamic,但发现一个明显的用例没有得到很好的实现。我正在尝试为面向对象的数据库创建便利包装器。它遭受了转换和可用性问题,因为它序列化和反序列化对象(它的方法返回Object)。

第一个问题: 我的数据库的get方法反序列化A类型的对象。Althought A有一个方法a(),我可能知道在给定的时刻,我有一个A,我不能调用a(),因为java只看到了Object 。如果“底层”对象实际上具有该方法,那么动态机制会为我设置,还是我必须自己在applyDynamic中处理它?我在REPL中尝试过,似乎对我没那么做。 如果我必须检查自己,最简单的方法是什么(使用scala的新反射),检查该对象是否具有方法“methodname”,如果是,则调用它。

https://stackoverflow.com/a/11056410/703862吓跑了 我回到java的反射,它非常容易地执行方法调用部分。

我想出了这个:

 scala> import java.lang.reflect.Method
 import java.lang.reflect.Method

 scala> class DynTest3 extends Dynamic {
 | def pee():String = "yuppie"
 |   def execSucced(method: Method, args: Any*):Boolean = return try{method.invoke(args); true} catch { case e: Exception => false }
 | def applyDynamic(methodName : String)(args: Any*){
 |     val succed = classOf[DynTest3].getDeclaredMethods.filter(m => m.getName() == methodName).exists(m => execSucced(m))
 | if (!succed)
 | throw new java.lang.NoSuchMethodException
 | }
 | }
 defined class DynTest3

但:

 scala> val y: Object = new DynTest3
 y: Object = DynTest3@74c74b55

 scala> y.asInstanceOf[Dynamic].pee()
 <console>:11: error: value applyDynamic is not a member of Dynamic
          y.asInstanceOf[Dynamic].pee()

所以基本上,这不起作用,即使我转向动态。 Casting to Dynamic已经使整个事情变得毫无用处,因为我想让用户免于投射。但也许有人可以创建一个隐式转换any2Dynamic ......

任何提示?

1 个答案:

答案 0 :(得分:6)

Scala中的

Dynamic是一种简单的编译时重写方案。详细信息在SIP-17: Type Dynamic中提供,在这里我将尝试用不同的词语解释它们。

与C#不同,dynamic引入了元对象的整个基础结构,绑定器和调用点缓存+实现了一个在运行时服务重载的mini-C#编译器,在Scala中我们决定采用最简单的方法。

我已经提到了编译时重写,现在是时候详细说明了。有几个规则,但让我们来看看最重要的规则。

如果我们无法编译foo.bar(arg1 ... argN),并且foo的静态类型是Dynamic的子类型,则将调用重写为foo.applyDynamic(“bar”)(arg1。 .argN)。重写的表达式将像往常一样进行类型检查,就好像它是由程序员手动编写的。

y.asInstanceOf[Dynamic].pee()的情况下,调用的接收者y.asInstanceOf [Dynamic]肯定是一个子类型为Dynamic的静态类型,因此触发了重写。重写的结果是y.asInstanceOf[Dynamic].applyDynamic("pee")()。但是,当Scala尝试对此表达式进行类型检查时,它会失败,因为Dynamic只是一个空标记特征,它没有定义任何方法。

要解决这个问题,你会转换为子类型Dynamic并拥有applyDynamic成员的东西(或者你没有强制转换,但是让你的API返回不是Object,而是那个东西)。您可以自己编写此内容或使用DynamicProxy(子类化以合并您的自定义逻辑),我希望它将包含在2.10.0-M5中。