AspectJ
2021-03-05 09:27
                         标签:override   类型   run   eth   名称空间   class   ica   str   值类型    AspectJ 3) 配置 当Spring IOC容器侦测到bean配置文件中的元素时,会自动为 与AspectJ切面匹配的bean创建代理  用AspectJ注解声明切面   AspectJ 标签:override   类型   run   eth   名称空间   class   ica   str   值类型    原文地址:https://www.cnblogs.com/lemonzhang/p/12908862.html
1、 简介
  AspectJ:Java社区里最完整最流行的AOP框架。(在Spring中AOP是一种思想,而AspectJ是一种AOP的更明确具体实现)
  在Spring2.0以上版本中,可以使用基于AspectJ注解或基于XML配置的AOP。
2、在Spring中启用AspectJ注解支持
1) 导入JAR包
? com.springsource.net.sf.cglib-2.2.0.jar
? com.springsource.org.aopalliance-1.0.0.jar
? com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar 
? spring-aop-4.0.0.RELEASE.jar
? spring-aspects-4.0.0.RELEASE.jar
2) 引入aop名称空间
 
    aop:aspectj-autoproxy />
1) 要在Spring中声明AspectJ切面,只需要在IOC容器中将切面声明为bean实例。
2) 当在Spring IOC容器中初始化AspectJ切面之后,Spring IOC容器就会为那些与 AspectJ切面相匹配的bean创建代理。
3) 在AspectJ注解中,切面只是一个带有@Aspect注解的Java类,它往往要包含很多通知。
4) 通知是标注有某种注解的简单的Java方法。
5) AspectJ支持5种类型的通知注解:
  ① @Before:前置通知,在方法执行之前执行
  ② @After:后置通知,在方法执行之后执行
  ③ @AfterRunning:返回通知,在方法返回结果之后执行
  ④ @AfterThrowing:异常通知,在方法抛出异常之后执行
  ⑤ @Around:环绕通知,围绕着方法执行package com.atguigu.spring.aop;
import java.util.Arrays;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
@Component
@Aspect//标注当前类为切面
public class MyloggerAspect {
    
    /**
     * @Before:将方法指定为前置通知
     * 必须设置value,其值为切入点表达式:找到切面作用到的连接点
     * 前置通知:作用于方法执行之前
     * @Before(value="execution(* com.atguigu.spring.aop.*.*(..))")
     * 第一个*代表任意的访问修饰符和返回值类型
     * 第二个*代表任意类
     * 第三个*代表类中任意方法
     * ..代表任意的参数列表
     */
    @Before(value = "execution(public int com.atguigu.spring.aop.MathImpl.add(int, int))")
    public void beforeMethod(JoinPoint joinPoint) {
        Object[] args = joinPoint.getArgs();//获取方法的参数
        String methodName = joinPoint.getSignature().getName();//获取方法名
        System.out.println("method:"+methodName+",arguments:"+Arrays.toString(args));
    }
    
    /**
     * @After:将方法标注为后置通知
     * 后置通知:作用于方法的finally语句块,即不管有没有异常都会执行
     */
    @After(value="execution(* com.atguigu.spring.aop.*.*(..))")
    public void afterMethod() {
        System.out.println("后置通知");
    }
    
    /**
     * @AfterReturning:将方法标注为返回通知
     * 返回通知:作用于方法执行之后
     * 可通过returning设置接收方法返回值的变量名
     * 要想在方法中使用,必须在方法的形参中设置和变量名相同的参数名的参数
     */
    @AfterReturning(value="execution(* com.atguigu.spring.aop.*.*(..))", returning="result")
    public void afterReturningMethod(JoinPoint joinPoint, Object result) {
        String methodName = joinPoint.getSignature().getName();
        System.out.println("method:"+methodName+",result:"+result);
    }
    
    /**
     * @AfterThrowing:将方法标注为异常通知(例外通知)
     * 异常通知(例外通知):作用于方法抛出异常时
     * 可通过throwing设置接收方法返回的异常信息
     * 在参数列表中课通过具体的异常类型,来对指定的异常信息进行操作
     */
    @AfterThrowing(value="execution(* com.atguigu.spring.aop.*.*(..))", throwing="ex")
    public void afterThrowingMethod(ArithmeticException ex) {
        System.out.println("有异常了,messages:"+ex);
    }
    
    //环绕通知
    @Around(value="execution(* com.atguigu.spring.aop.*.*(..))")
    public Object aroundMethod(ProceedingJoinPoint joinPoint) {
        
        Object result = null;
        
        try {
            //前置通知
            System.out.println("前置通知");
            result = joinPoint.proceed();//执行方法
            //返回通知
            System.out.println("返回通知");
            return result;
        } catch (Throwable e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
            //异常通知
            System.out.println("异常通知");
        } finally {
            //后置通知
            System.out.println("后置通知");
        }
        
        return -1;
    }
}
package com.atguigu.spring.aop;
import org.springframework.stereotype.Component;
@Component   //都应该交由Spring处理
public class MathImpl implements MathI {
    @Override
    public int add(int i, int j) {
        int result = i + j;
        return result;
    }
    @Override
    public int sub(int i, int j) {
        int result = i - j;
        return result;
    }
    @Override
    public int mul(int i, int j) {
        int result = i * j;
        return result;
    }
    @Override
    public int div(int i, int j) {
        int result = i / j;
        return result;
    }
    
    
}
package com.atguigu.spring.aop;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
    public static void main(String[] args) {
        
        ApplicationContext ac = new ClassPathXmlApplicationContext("aop.xml");
        MathI math = ac.getBean("mathImpl", MathI.class);
        System.out.println(math.getClass().getName());
        int i = math.div(4, 1);
        System.out.println(i);
        
    }
    
}