加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 大数据 > 正文

反射 – Groovy:在调用任何方法之前和之后,执行代码是透明的

发布时间:2020-12-14 16:23:22 所属栏目:大数据 来源:网络整理
导读:假设我们有一个带有一些方法(静态或非静态)的groovy类. 我想要做的是在调用此类的每个方法之前和之后执行一些代码,而根本不需要触及类,也不需要动态操作每个方法内部的代码. 我尝试使用groovy metaClass;获取metaClass的所有方法,然后使用包装方法动态替换ev
假设我们有一个带有一些方法(静态或非静态)的groovy类.
我想要做的是在调用此类的每个方法之前和之后执行一些代码,而根本不需要触及类,也不需要动态操作每个方法内部的代码.

我尝试使用groovy metaClass;获取metaClass的所有方法,然后使用包装方法动态替换every方法,包含一些代码,并在中间调用旧方法.问题是,我不知道每个原始方法的参数,所以我不能用新方法(闭包)替换旧方法,因为我不能动态地创建具有不同数量和类型的参数的包装闭包,即使yi可以,我不知道如何在包装封口内访问它们.我需要包装闭包具有与旧方法相同的签名,以便在有人在透明地更改类之后尝试调用旧方法时调用闭包.

在Javascript中,例如我可以使用args []数组访问函数体中的所有参数,即使我在编写代码时不知道参数名称.

我怎么能在groovy做到这一点?或者他们可能是另一种方式来实现我的目标?

解决方法

像下面的东西会做什么?使用invokeMethod拦截对每个方法的调用.测试是自我解释的.

说明:

下面的metaClass实现从GroovyObject开始覆盖invokeMethod.由于所有groovy对象都继承自GroovyObject,因此我们可以灵活地操作/拦截方法调用,甚至可以指定我们自己的methodMissing实现.我们需要一个覆盖静态方法和一个非静态方法.基本上,invokeMethod拦截对定义它的Class上的每个方法的调用.我们最终得到了每种方法的一些前后功能.使用反射,下面的实现能够通过其名称和参数找出方法,并在运行时调用它.

注意:-

>确保从闭包中返回方法调用的返回值
>可能是一个昂贵的类有很多方法与繁重的实现
>如果需要选择性执行,则检查方法名称,然后拦截呼叫

执行:

class Dummy {
    def method1() { System.out.println "In method 1" }
    def method2(String str) { System.out.println "In method 2" }
    static def method3(int a,int b) { 
        System.out.println "In static method 3" 
    }    
}

Dummy.metaClass.invokeMethod = {String name,args ->
   def result

   System.out.println(
     "Do something before $name is called with args $args")

   try {
      result = delegate.metaClass.getMetaMethod(name,args)
                                 .invoke(delegate,args)
   } catch(Exception e) {
       System.out.println "Handling exception for method $name"
   }

   System.out.println(
      "Do something after $name was called with args $args n")

   result
}

Dummy.metaClass.'static'.invokeMethod = {String name,args ->
   def result

   System.out.println(
     "Do something before static method $name is called with args $args")

   try {
      result = delegate.metaClass.getMetaMethod(name,args)
   } catch(Exception e) {
       System.out.println "Handling exception for method $name"
   }

   System.out.println(
     "Do something after static method $name was called with args $args n")

   result
}

def dummy = new Dummy()
dummy.method1()
dummy.method2('Test')
Dummy.method3(1,2)

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读