| Spring 框架学习(八)——AOP的认识与使用
 一、Spring AOP 常用概念及铺垫
 先说几个常用的概念以及铺垫的知识 
 横切点:就是我们要给方法前加一个打印日志的功能,或者先校验等等,这些功能 切面(Aspect):将横切关注点进行模块化的一个类 通知(Adviser):就是切面里面具体的方法,用来切入到具体位置的方法里面 切入点(PointCut):一个业务类中的具体方法,就是把通知切入到这个方法里面 连接点(JoinPoint):通过连接点我们可以知道切入点方法对象的很多信息。 
 
 二、通知的介绍
 通知类型就是想要加的代码(校验、日志等) 是在对象方法的前面还是后面执行的类型,这就是通知类型。 
 (1)before 前置通知在方法执行前执行 如果方法出现了异常,不会影响前置通知的执行 应用:通常应用在执行方法前各种校验 
 (2)after 后置通知在方法执行完毕之后执行 无论方法是否出现异常终止都会执行 应用:清理现场,关闭、释放资源 
 (3)around 环绕通知方法执行前后分别执行 如果方法中出现异常终止,那么末尾的通知就不执行了 应用:各个方面,结合了前置和后置的优点,还可以拿到对象方法的各种参数及信息 
 (4)afterReturning 返回后通知方法正常返回后执行 如果方法抛出异常,那么无法执行 应用:常规结果数据处理 
 (5)afterThrowing 抛出异常后通知方法抛出异常后执行 如果方法没有抛出异常,无法执行 应用:抛出异常后对信息进行包装 
 三、切点表达式说明
 (1)作用
 用来匹配具体切入点(方法)的具体位置 
 (2)格式
 execution(切点表达式) execution([修饰符] 返回类型 包名.类名.方法名(参数类型)) 方法访问修饰符可以进行省略,但是后面的东西必须得有 
 (3)通配符介绍具体使用 我们想要把切点定义到 com.bit.service 包下的 UserService 类中的 void add(int a) 方法 有很多种写法 
 1.每一部分都写的具体   execution("void com.bit.service.UserService.add(int)")
 
 2.完全使用通配符   execution("* *..service.*(..)")
  
  
  
  
  
  
   execution("* *..*.*(..)")
  
  
  
  
  
  
   execution("* *..*(..)")
  
  
  
  
  
 四、连接点及环绕方法参数使用
 前置通知,写一个方法,在切入点之前执行通知即可 后置通知,写一个方法,在切入点之前执行通知即可 那么环绕通知的代码如何进行环绕呢? 这里就要用到 连接点(JoinPoint) 的一些使用了 
 (1) JoinPoint 作为环绕通知 的方法参数
 JoinPoint对象封装了SpringAop中切面方法的信息,在切面方法中添加JoinPoint参数,就可以获取到封装了该方法信息的JoinPoint对象. JoinPoint 只有获取切入点对象方法信息的一些功能,不能帮助我们进行环绕写代码 
 能够获取对象方法参数、方法签名等信息 
 (2) ProceedingJoinPoint 作为环绕通知 的方法参数
 ProceedingJoinPoint对象是JoinPoint的子接口,该对象只用在@Around的切面方法中,添加了
  
 Object proceed() throws Throwable //执行目标方法  
 Object proceed(Object[] var1) throws Throwable //传入的新的参数去执行目标方法两个方法.
 
 调用proceed方法,相当于执行切入点方法 
 (3) 环绕通知的写法    public void around(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("Around=====方法环绕前====");
        joinPoint.proceed();  
        System.out.println("Around======方法环绕后=====");
    }
 
 (4) 在执行方法的时候修改方法参数
 1、使用 Object [] args 接收原方法传入的参数  Object[] args = joinPoint.getArgs();
 
 2、对数组中的参数进行修改   for (int i = 0; i <args.length ; i++) {
        args[i] = (Integer)args[i]+10;
  }
 
 3、执行proceed带参数的方法,将修改过的object[] args进行传入 joinPoint.proceed(args);  
 
 五、Spring-AOP 准备工作
 (1)加入aspect织入包
 在编写SpringAOP面向切面编程时,需要导入一个aspectjweaver.jar的包,它的主要作用是负责解析切入点表达式。     
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.9.1</version>
        </dependency>
 
 (2)加入aop约束以及开启aop注解支持
 可以在xml使用<aop:config 回车自动生成,下面是aop约束已经加上了的 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
    
    
    <aop:aspectj-autoproxy />
    
    
    <context:annotation-config/>
    <context:component-scan base-package="com.*"/>
</beans>
 
 六、Spring中如何使用AOP
 (1)xml配置使用AOP
 (1)先写一个业务的接口 package com.service;
public interface UserService {
    void add();
};
 
 (2)给这个接口创建创建一个实现类,类中的方法作为切入点。 package com.service;
import org.springframework.stereotype.Service;
@Service
public class UserServiceImpl implements UserService {
    public void add() {
        System.out.println("执行了add方法");
    }
}
 
 (3)自定义一个类作为切面,在里面实现一些方法作为具体的通知。 package com.config;
import org.aspectj.lang.ProceedingJoinPoint;
public class DiyAspect {
    
     public void before(){
         System.out.println("Before=====方法执行前====");
     }
    public void after(){
        System.out.println("After=====方法执行后====");
    }
    public void around(ProceedingJoinPoint pj) throws Throwable {
        System.out.println("Around=====方法环绕前====");
        pj.proceed();  
        System.out.println("Around======方法环绕后=====");
    }
    
}
 
 (4)通过xml配置定义切点、切面、通知,以及使得切面里的通知绑定到切点上。 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
    <bean id="diyAspect" class="com.config.DiyAspect"/>
    
    
    <aop:aspectj-autoproxy />
    
    <aop:config >
        <aop:pointcut id="pointcut" expression="execution(* *..service.*.*(..))"/>
        <aop:aspect id="diy" ref="diyAspect">
            <aop:before method="before" pointcut-ref="pointcut"/>
            <aop:around method="around" pointcut-ref="pointcut"/>
            <aop:after method="after" pointcut-ref="pointcut"/>
        </aop:aspect>
    </aop:config>
    <context:annotation-config/>
    <context:component-scan base-package="com.*"/>
</beans>
 
 (5)进行测试,执行add方法,查看通知增强是否绑定切入点成功 package com.controller;
import com.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.stereotype.Controller;
@Controller
public class UserController {
    public static void main(String[] args) {
        ApplicationContext context =new ClassPathXmlApplicationContext("spring-config.xml");
        UserService userService = context.getBean("userServiceImpl", UserService.class);
        userService.add();
    }
}
 
 (6)我们可以确定,通过xml注解的方式,前置通知、后置通知、环绕通知的执行顺序 
 环绕前和环绕后被 before after包裹着 
 (2)注解开发使用AOP
 (1)使用注解 @Aspect 将自定义类作为切面,@PointCut 定义切点的位置,@Before / @After 定义通知作用到切点上 package com.config;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class DiyAspect {
    
    
    @Pointcut("execution( * com.service.UserServiceImpl.add(..))")
    public void pointCut(){
    }
    @Before("pointCut()")
     public void before(){
         System.out.println("Before=====方法执行前====");
     }
     @After("pointCut()")
    public void after(){
        System.out.println("After=====方法执行后====");
    }
    @Around("pointCut()")
    public void around(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("Around=====方法环绕前====");
        joinPoint.proceed();  
        System.out.println("Around======方法环绕后=====");
    }
}
 (2)xml文件中可以完全不同写aop:config 的配置内容,但是必须加上aop约束和aop注解支持 <?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
    
    <aop:aspectj-autoproxy />
    <context:annotation-config/>
    <context:component-scan base-package="com.*"/>
</beans>
 
 (3)进行测试,执行add方法,查看通知增强是否绑定切入点成功 package com.controller;
import com.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.stereotype.Controller;
@Controller
public class UserController {
    public static void main(String[] args) {
        ApplicationContext context =new ClassPathXmlApplicationContext("spring-config.xml");
        UserService userService = context.getBean("userServiceImpl", UserService.class);
        userService.add();
    }
}
 
 (4) 我们可以确定,通过注解开发的方式,前置通知、后置通知、环绕通知的执行顺序和之前xml配置就不一样了 
 环绕通知在 before 和 after的外面包裹 |