06 Spring AOP

1、AOP概念

AOP就是面向切面编程,在原有方法的前边或者后边增加一些业务逻辑代码。

相关术语:

1、连接点:程序执行过程中的某个时间点。

2、切入点:就是需要切面编程的连接点。

3、通知:是在切入点前边或者后边要执行的一段代码,通知有分为:前置通知,后置通知,异常通知,最终通知和环绕通知等。

4、切面:指封装切入点和通知的一个类。

5、目标对象:被增强的对象。

2、通知类型说明

1、前置通知:在目标方法执行前实施增强。

2、后置返回通知:在目标方法执行后执行,如果产生了异常,则不会被执行。

3、后置最终通知:在目标方法执行后执行,有异常,也会被执行的通知。

4、异常通知:当目标方法执行过程中产生异常,则执行的通知。

5、环绕通知:在目标方法执行前和目标方法执行后执行的通知。

6、引入通知:在目标类中增加一些新的方法和属性,用来对目标类进行扩展。

3、AspectJ切入表表达式

基本语法:

@Pointcut(value="execution(modifiers-pattern? ret-type-pattern declaring-type-pattern.methodName-pattern(param-pattern) throws-pattern?)")

1)末尾有?的代表可以省略。

2)modifiers-pattern? 目标方法的修饰符匹配符,如public , private

3)ret-type-pattern 目标方法返回值匹配符,如void、string等,* 代表所有类型。

4)declaring-type-pattern 目标方法类路径匹配符,如cn.sjxy.service.UserService,*代表所有类型

5)methodName-pattern 目标方法名称匹配符,例如Insert,Update,*代表所有方法名。

6)param-pattern 目标方法参数匹配符,(..)代表所有的参数,包含无参方法。

7)throws-pattern? 目标方法抛出的异常类型

例如:

//定义切入点,设定切入点表达式,方法名是切入点名称
@Pointcut(value="execution(* cn.sjxy.Service.ProductService.*(..))")
@Pointcut(value="execution(* cn.sjxy.Service.ProductService.addProduct(..))")

4、AspectJ框架的使用 - 注解

1、pom.xml引入依赖包

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>
  <groupId>cn.sjxy</groupId>
  <artifactId>chapter02Maven</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  
  <dependencies>
  
  	<!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
  	<dependency>
    	<groupId>org.springframework</groupId>
    	<artifactId>spring-context</artifactId>
    	<version>5.3.0</version>
	</dependency>
	
	<!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
	<dependency>
    	<groupId>org.springframework</groupId>
    	<artifactId>spring-aop</artifactId>
    	<version>5.3.0</version>
	</dependency>
    
    <!-- https://mvnrepository.com/artifact/org.springframework/spring-aspects -->
	<dependency>
    	<groupId>org.springframework</groupId>
    	<artifactId>spring-aspects</artifactId>
    	<version>5.3.0</version>
	</dependency>
	
  </dependencies>
</project>

2、applicationContext.xml 开启AspectJ注解使用方式

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop            
           http://www.springframework.org/schema/aop/spring-aop-3.1.xsd">

	<!-- 开启扫包 -->
    <context:component-scan base-package="cn.sjxy"/>
	
	<!-- 开启AspectJ 注解 -->
    <aop:aspectj-autoproxy />
    
</beans>

3、业务类代码

package cn.sjxy.Service;

import org.springframework.stereotype.Service;

@Service(value = "productService")
public class ProductService {
	
	public void addProduct() {
		System.out.println(">>>添加商品");
		int a=1,b=0,c;
		c=a/b;
	}
	
	public String findProduct()
	{
		System.out.println(">>>查看商品");
		return "SSM框架";
	}
	
	public void modifyProduct()
	{
		System.out.println(">>>修改商品");
	}
	
	public void deleteProduct()
	{
		System.out.println(">>>删除商品");
	}
}

4、切面类代码

package cn.sjxy.Aspect;

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.stereotype.Component;

@Component
@Aspect
public class MyAnnotationAspect {
	
	//定义切入点,设定切入点表达式,方法名是切入点名称
	@Pointcut(value="execution(* cn.sjxy.Service.ProductService.addProduct(..))")
	private void myPointCut() {}
	
	//前置通知
	@Before(value="myPointCut()")
	public void beforeAdvice(JoinPoint jp)
	{
		System.out.println("2前置通知");
		System.out.println("被通知的方法名:"+jp.getSignature().getName());
	}
	
	//后置返回通知,能够获取到方法的返回值,有异常,则不执行
	@AfterReturning(value="myPointCut()",returning = "obj")
	public void afterReturningAdvice(JoinPoint jp,Object obj)
	{
		System.out.println("3后置返回通知");
		System.out.println("被通知的方法名:"+jp.getSignature().getName());
		System.out.println("方法返回结果:"+obj);
	}
	
	//环绕通知,在方法的前边和后边执行,有异常,则后边不再执行
	@Around(value="myPointCut()")
	public Object roundAdvice(ProceedingJoinPoint pjp) throws Throwable{

		System.out.println("1=======begin========");
		Object resultValue = pjp.proceed();
		System.out.println("6=======end========");
		return resultValue;
	}
	
	//异常通知,产生异常时执行
	@AfterThrowing(pointcut="myPointCut()",throwing="ex")
	public void exceptionAdvice(Throwable ex)
	{
		System.out.println("4异常通知:程序的运行异常为:"+ex.getMessage());
	}
	
	//最终通知,有异常也执行
	@After("myPointCut()")
	public void afterAdvice()
	{
		System.out.println("5最终通知,产生异常也执行");
	}
}

5、测试类代码:

public class SpringTest {
	public static void main(String[] args)
	{
		 ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
         
		 ProductService s = (ProductService)context.getBean("productService");
		 s.addProduct();	 
	}
}

6、执行结果截图 - 无异常

7、执行结果截图 - 有异常

 

3和6的通知不会被执行到。

 

posted @ 2022-01-20 20:20  草莓爸  阅读(31)  评论(0编辑  收藏  举报