zoukankan      html  css  js  c++  java
  • ProceedingJoinPoint获取当前方法

    aspectJ切面通过ProceedingJoinPoint想要获取当前执行的方法:

    错误方法:

         Signature s = pjp.getSignature();
        MethodSignature ms = (MethodSignature)s;
        Method m = ms.getMethod();

    这种方式获取到的方法是接口的方法而不是具体的实现类的方法,因此是错误的。

    正确方法:

            Signature sig = pjp.getSignature();
            MethodSignature msig = null;
            if (!(sig instanceof MethodSignature)) {
                throw new IllegalArgumentException("该注解只能用于方法");
            }
            msig = (MethodSignature) sig;
            Object target = pjp.getTarget();
            Method currentMethod = target.getClass().getMethod(msig.getName(), msig.getParameterTypes());

    AspectJ使用org.aspectj.lang.JoinPoint接口表示目标类连接点对象,如果是环绕增强时,使用org.aspectj.lang.ProceedingJoinPoint表示连接点对象,该类是JoinPoint的子接口。任何一个增强方法都可以通过将第一个入参声明为JoinPoint访问到连接点上下文的信息。我们先来了解一下这两个接口的主要方法: 
    1)JoinPoint 
       java.lang.Object[] getArgs():获取连接点方法运行时的入参列表; 
       Signature getSignature() :获取连接点的方法签名对象; 
       java.lang.Object getTarget() :获取连接点所在的目标对象; 
       java.lang.Object getThis() :获取代理对象本身; 
    2)ProceedingJoinPoint 
    ProceedingJoinPoint继承JoinPoint子接口,它新增了两个用于执行连接点方法的方法: 
       java.lang.Object proceed() throws java.lang.Throwable:通过反射执行目标对象的连接点处的方法; 
       java.lang.Object proceed(java.lang.Object[] args) throws java.lang.Throwable:通过反射执行目标对象连接点处的方法,不过使用新的入参替换原来的入参。 

    JoinPoint 对象

    JoinPoint对象封装了SpringAop中切面方法的信息,在切面方法中添加JoinPoint参数,就可以获取到封装了该方法信息的JoinPoint对象. 
    常用api:

    方法名功能
    Signature getSignature(); 获取封装了署名信息的对象,在该对象中可以获取到目标方法名,所属类的Class等信息
    Object[] getArgs(); 获取传入目标方法的参数对象
    Object getTarget(); 获取被代理的对象
    Object getThis(); 获取代理对象

    ProceedingJoinPoint对象

    ProceedingJoinPoint对象是JoinPoint的子接口,该对象只用在@Around的切面方法中, 
    添加了 
    Object proceed() throws Throwable //执行目标方法 
    Object proceed(Object[] var1) throws Throwable //传入的新的参数去执行目标方法 
    两个方法.

    Demo

    切面类

    @Aspect
    @Component
    public class aopAspect {
        /**
         * 定义一个切入点表达式,用来确定哪些类需要代理
         * execution(* aopdemo.*.*(..))代表aopdemo包下所有类的所有方法都会被代理
         */
        @Pointcut("execution(* aopdemo.*.*(..))")
        public void declareJoinPointerExpression() {}
     
        /**
         * 前置方法,在目标方法执行前执行
         * @param joinPoint 封装了代理方法信息的对象,若用不到则可以忽略不写
         */
        @Before("declareJoinPointerExpression()")
        public void beforeMethod(JoinPoint joinPoint){
            System.out.println("目标方法名为:" + joinPoint.getSignature().getName());
            System.out.println("目标方法所属类的简单类名:" +        joinPoint.getSignature().getDeclaringType().getSimpleName());
            System.out.println("目标方法所属类的类名:" + joinPoint.getSignature().getDeclaringTypeName());
            System.out.println("目标方法声明类型:" + Modifier.toString(joinPoint.getSignature().getModifiers()));
            //获取传入目标方法的参数
            Object[] args = joinPoint.getArgs();
            for (int i = 0; i < args.length; i++) {
                System.out.println("第" + (i+1) + "个参数为:" + args[i]);
            }
            System.out.println("被代理的对象:" + joinPoint.getTarget());
            System.out.println("代理对象自己:" + joinPoint.getThis());
        }
     
        /**
         * 环绕方法,可自定义目标方法执行的时机
         * @param pjd JoinPoint的子接口,添加了
         *            Object proceed() throws Throwable 执行目标方法
         *            Object proceed(Object[] var1) throws Throwable 传入的新的参数去执行目标方法
         *            两个方法
         * @return 此方法需要返回值,返回值视为目标方法的返回值
         */
        @Around("declareJoinPointerExpression()")
        public Object aroundMethod(ProceedingJoinPoint pjd){
            Object result = null;
     
            try {
                //前置通知
                System.out.println("目标方法执行前...");
                //执行目标方法
                //result = pjd.proeed();
                //用新的参数值执行目标方法
                result = pjd.proceed(new Object[]{"newSpring","newAop"});
                //返回通知
                System.out.println("目标方法返回结果后...");
            } catch (Throwable e) {
                //异常通知
                System.out.println("执行目标方法异常后...");
                throw new RuntimeException(e);
            }
            //后置通知
            System.out.println("目标方法执行后...");
     
            return result;
        }
    }

    被代理类

    /**
     * 被代理对象
     */
    @Component
    public class TargetClass {
        /**
         * 拼接两个字符串
         */
        public String joint(String str1, String str2) {
            return str1 + "+" + str2;
        }
    }

    测试类

    public class TestAop {
        @Test
        public void testAOP() {
            //1、创建Spring的IOC的容器
            ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:bean.xml");
     
            //2、从IOC容器中获取bean的实例
            TargetClass targetClass = (TargetClass) ctx.getBean("targetClass");
     
            //3、使用bean
            String result = targetClass.joint("spring","aop");
            System.out.println("result:" + result);
        }
    }

    输出结果

    目标方法执行前...
    目标方法名为:joint
    目标方法所属类的简单类名:TargetClass
    目标方法所属类的类名:aopdemo.TargetClass
    目标方法声明类型:public
    第1个参数为:newSpring
    第2个参数为:newAop
    被代理的对象:aopdemo.TargetClass@4efc180e
    代理对象自己:aopdemo.TargetClass@4efc180e
    目标方法返回结果后...
    目标方法执行后...
    result:newSpring+newAop

    参考文章: 
    http://blog.csdn.net/ochangwen/article/details/52557724 
    http://blog.csdn.net/a9529lty/article/details/7031070

  • 相关阅读:
    jquery获取tr并更改tr内容
    jquery获取元素索引值index()
    禁止apache显示目录索引 apache禁止列目录
    mysql启动错误之mysql启动报1067错误如何解决
    Expo大作战(四)--快速用expo构建一个app,expo中的关键术语
    Expo大作战(三)--针对已经开发过react native项目开发人员有针对性的介绍了expo,expo的局限性,开发时项目选型注意点等
    Expo大作战(二)--expo的生命周期,expo社区交流方式,expo学习必备资源,开发使用expo时关注的一些问题
    Expo大作战(一)--什么是expo,如何安装expo clinet和xde,xde如何使用
    Linux(CentOS)之-性能监控
    [转]winform程序textbox滚动条保持在最下面 内容不闪烁
  • 原文地址:https://www.cnblogs.com/muxi0407/p/11818999.html
Copyright © 2011-2022 走看看