《Pro Spring》学习笔记之DynamicMethodMatcherPointCut动态切入点实例

本文介绍了如何在Spring中使用DynamicMethodMatcherPointCut实现动态切入点,通过示例展示了如何根据方法参数决定是否执行通知,从而提高效率并避免潜在的异常。文章包括BeanOne.java的代码,环绕通知的配置,以及测试代码的运行结果分析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

如果我们需要根据方法的参数不同而决定事都进行切入,而不是不管方法是什么参数全部切入的话,我们就需要使用动态切入点,在每次方法调用时候都进行检查(确切的说是参数检查),;来决定是否执行通知

 

BeanOne.java

 

package DynamicPointCutAdvisor;

public class BeanOne {
   
public void foo(){
       System.out.println(
"foo-one");
   }

   
public void bar(){
       System.out.println(
"bar-one");
   }

   
public void dynamic(int x){
       System.out.println(x);
   }

}

 动态切入点:

 

package DynamicPointCutAdvisor;

import java.lang.reflect.Method;

import org.springframework.aop.ClassFilter;
import org.springframework.aop.support.DynamicMethodMatcherPointcut;

public class SimpleDynamicPointcut extends DynamicMethodMatcherPointcut {

    
    
//静态方法检查,为避免每次调用bar方法是都进行一次检测,导致效率降低
    public boolean matches(Method method, Class cls) {
        
return "dynamic".equals(method.getName());
    }

    
//当参数不为100时候,进行切入
    public boolean matches(Method method, Class cls, Object[] obj) {
        
int x=((Integer)obj[0]).intValue();
        
return (x!=100);
    }

    
public ClassFilter getClassFilter(){
        
return new ClassFilter(){
            
public boolean matches(Class cls){
                
return cls==BeanOne.class;
            }

        }
;
    }


}

 

环绕通知:

 

package DynamicPointCutAdvisor;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

public class SimpleAdvise implements MethodInterceptor {

    
public Object invoke(MethodInvocation invocation) throws Throwable {
        System.out.println(
"before");
        Object retVal
=invocation.proceed();
        System.out.println(
"after");
        
return retVal;
    }


}

 

测试代码:

 

package DynamicPointCutAdvisor;

import org.aopalliance.aop.Advice;
import org.springframework.aop.Advisor;
import org.springframework.aop.Pointcut;
import org.springframework.aop.framework.ProxyFactory;
import org.springframework.aop.support.DefaultPointcutAdvisor;

public class Test {

    
/**
     * 
@param args
     
*/

    
public static void main(String[] args) {
         BeanOne one
=new BeanOne();
        
         BeanOne proxyOne;
    
         
         Pointcut pc
=new SimpleDynamicPointcut();
         Advice advice
=new SimpleAdvise();
         Advisor advisor
=new DefaultPointcutAdvisor(pc,advice);
         
         
         
//创建BeanOne代理
         ProxyFactory pf1=new ProxyFactory();
         pf1.addAdvisor(advisor);
         pf1.setTarget(one);
         proxyOne
=(BeanOne)pf1.getProxy();
         
     
         
         
         proxyOne.dynamic(
100);
         System.out.println(
"-------------------");
         proxyOne.dynamic(
101);
         System.out.println(
"-------------------");
         proxyOne.bar();

    }


}

 

运行结果:

100
-------------------
before
101
after

-------------------
bar-one

 

后记

如果我们在切入点类中的方法检查代码中加入提示,如:

 

package DynamicPointCutAdvisor;

import java.lang.reflect.Method;

import org.springframework.aop.ClassFilter;
import org.springframework.aop.support.DynamicMethodMatcherPointcut;

public class SimpleDynamicPointcut extends DynamicMethodMatcherPointcut {

    
    
//静态方法检查,为避免每次调用bar方法是都进行一次检测,导致效率降低
    public boolean matches(Method method, Class cls) {
        System.out.println(
"static check for "+method.getName());
        
return "dynamic".equals(method.getName());
    }

    
//当参数不为100时候,进行切入
    public boolean matches(Method method, Class cls, Object[] obj) {
        System.out.println(
"dynamic check for "+method.getName());
        
int x=((Integer)obj[0]).intValue();
        
return (x!=100);
    }

    
public ClassFilter getClassFilter(){
        
return new ClassFilter(){
            
public boolean matches(Class cls){
                
return cls==BeanOne.class;
            }

        }
;
    }


}


可以看到结果如下:

static check for dynamic
static check for bar
static check for foo
static check for clone
static check for toString
static check for dynamic

dynamic check for dynamic
100
-------------------
dynamic check for dynamic
before
101
after
-------------------
static check for bar
bar-one

可以看到,红色部分代表静态方法类型检查,绿色代表动态方法类型检查

当我们实现了静态检查|(public boolean matches(Method method, Class cls) )时候,动态类型对不匹配的方法,如本例中的bar不再进行动态检查,提高了效率,如果没有public boolean matches(Method method, Class cls) 方法,则所有方法势必都会进行类型检查,不仅降低效率,还有可能因为方法参数的不同导致异常(如bar是没有参数的,使用动态检查中的 int x=((Integer)obj[0]).intValue();就会出现异常)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值