-
AOP(Aspect-Oriented Programming,面向切面编程):是一种新的方法论,是对传 统 OOP(Object-Oriented Programming,面向对象编程)的补充。
-
AOP实现的关键在于AOP框架自动创建的AOP代理,AOP代理主要分为静态代理和动态代理 以AspectJ为代表的静态代理。 以Spring AOP为代表的动态代理。
-
AOP编程操作的主要对象是切面(aspect),而切面模块化横切关注点。
-
在应用AOP编程时,仍然需要定义公共功能,但可以明确的定义这个功能应用在哪里,以什么方式应用,并且不必修改受影响的类。这样一来横切关注点就被模块化到特殊的类里——这样的类我们通常称之为“切面”。
-
AOP的好处:
-
每个事物逻辑位于一个位置,代码不分散,便于维护和升级
-
业务模块更简洁,只包含核心业务代码
-
AOP图解
-
从每个方法中抽取出来的同一类非核心业务(打印日志业务)。
2.2 切面(Aspect)
封装横切关注点信息的类,每个关注点体现为一个通知方法(比如参数校验的类,打印日志的类)。
2.3 通知(Advice)
切面必须要完成的各个具体工作(参数校验的、打印日志的方法)
2.4 目标(Target)
被通知的对象
2.5 代理(Proxy)
向目标对象应用的
2.6 连接点(Joinpoint)
横切关注点在程序代码中的具体体现,对应程序执行的某个特定位置。例如:类某个方法调用前、调用后、方法捕获到异常后等(比如日志代码的位置)。
在应用程序中可以使用横纵两个坐标来定位一个具体的连接点:
2.7 切入点(pointcut)
定位连接点的方式。每个类的方法中都包含多个连接点,所以连接点是类中客观存在的事物。如果把连接点看作数据库中的记录,那么切入点就是查询条件——AOP可以通过切入点定位到特定的连接点。切点通过org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件。
三、AspectJ
AspectJ:Java社区里最完整最流行的AOP框架。
在Spring2.0以上版本中,可以使用基于AspectJ注解或基于XML配置的AOP。
对加减乘除的日志打印用AOP实现
3.1、在Spring中启用AspectJ注解
<dependency> <groupId>cglib</groupId> <artifactId>cglib</artifactId> <version>2.2.2</version> </dependency> <dependency> <groupId>aopalliance</groupId> <artifactId>aopalliance</artifactId> <version>1.0</version> </dependency> <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver --> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.5</version> </dependency> <!-- https://mvnrepository.com/artifact/org.springframework/spring-aop --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>5.2.2.RELEASE</version> </dependency> <!-- https://mvnrepository.com/artifact/org.springframework/spring-aspects --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>5.2.2.RELEASE</version> </dependency>
-
-
配置
-
<aop:aspectj-autoproxy>
-
<context:component-scan base-package="com.jdy.spring2020"/> <!--基于注解开发AspectJ:主要作用是为切面中能作用到的目标类生成代理--> <aop:aspectj-autoproxy/>
3.2、用AspectJ注解声明切面
上面讲到切面必须要完成的各个具体工作叫做通知,可以知道一个切面中可以有很多个通知。
通知的分类
-
-
@Before:前置通知,在方法执行之前执行
-
@After:后置通知,在方法执行之后执行
-
@AfterRunning:返回通知,在方法返回结果之后执行
-
@AfterThrowing:异常通知,在方法抛出异常之后执行
-
@AfterThrowing:异常通知,在方法抛出异常之后执行
-
/** * @Aspect:表示是一个切面 */ @Component @Aspect @EnableAspectJAutoProxy(proxyTargetClass = true) public class LogAop { /** * 前置通知: * @Before("execution( com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))") * 在ArithmeticCalculatorImpl.add方法前之前前置通知 * */ @Before(value = "execution(* com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))") public void before(){ System.out.println("AO....前置通知"); } /** * * @param joinPoint:连接点对象 */ @AfterReturning(value = "execution(* com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))",returning ="result" ) public void after(JoinPoint joinPoint){ //方法的名字 String name = joinPoint.getSignature().getName(); System.out.println("后置通知方法名字:" + name); } /** * 返回通知:目标方法执行结束后,得到方法的返回值 * 获取方法的返回值:通过returnning来指定一个名字,必须要与方法的一个参数名成一致。 * @param joinPoint * @param result */ @AfterReturning(value = "execution(* com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))",returning ="result" ) public void afterRunning(JoinPoint joinPoint,Object result){ //方法的名字 String name = joinPoint.getSignature().getName(); System.out.println("返回通知方法名字:" + name); System.out.println("返回通知方法返回值:" + result); } /** * 异常通知:目标方法执行结束后,目标方法抛出异常 * 获取方法的异常:通过Throwing来指定一个名字,必须要与方法的一个参数名一致 * 可以通过形参中异常的类型来设置抛出指定异常才会执行异常通知 * @param joinPoint */ @AfterThrowing(value = "execution(public int com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.sub(int,int))",throwing = "ex") public void afterThrowing(JoinPoint joinPoint,Exception ex){ //方法的名字 String name = joinPoint.getSignature().getName(); System.out.println("异常通知方法名字:" + name); System.out.println("异常通知方法返回异常:" + ex); } //可以通过形参中异常的类型来设置抛出指定异常才会执行异常通知 @AfterThrowing(value = "execution(public int com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.*(int,int))",throwing = "ex") public void afterThrowing1(JoinPoint joinPoint,NullPointerException ex){ //方法的名字 String name = joinPoint.getSignature().getName(); System.out.println("异常通知方法名字" + name); System.out.println("异常通知方法返回异常" + ex); } /** * 环绕通知 * @param joinPoint */ @Around(value = "execution(public int com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))") public Object around(ProceedingJoinPoint joinPoint) { //方法的名字 Object obj = null; //前置 try { //执行目标方法,相当于动态代理的invoke方法 System.out.println("环绕通知---->方法名字" + joinPoint.getSignature().getName()); obj = joinPoint.proceed(); System.out.println("环绕通知---->方法proceed" + obj); } catch (Throwable throwable) { throwable.printStackTrace(); }finally { //后置 } return obj; } }
public interface ArithmeticCalculator { void add(int a,int b); }
@Service("arithmeticCalculatorImpl") public class ArithmeticCalculatorImpl implements ArithmeticCalculator { @Override public void add(int a, int b) { System.out.println("日志:The Method add begin ["+a+","+b+"]"); int result = a+b; System.out.println("result = " + result); } }
public class Test_03 { ClassPathXmlApplicationContext context = null; { context = new ClassPathXmlApplicationContext("application_03.xml"); } @Test public void test() { ArithmeticCalculatorImpl impl = context.getBean("arithmeticCalculatorImpl", ArithmeticCalculatorImpl.class); impl.add(2,3); } }