Spring 框架的 AOP

Spring AOP是什么?

Spring 框架的 AOP

Aspect Oriented Programming:面向切面编程

什么时候会出现面向切面编程的需求?按照软件重构的思想,如果多个类中出现重复的代码,就应该考虑定义一个共同的抽象类,将这些共同的代码提取到抽象类中,比如Teacher,Student都有username,那么就可以把username及相关的get、set方法抽取到SysUser中,这种情况,我们称为纵向抽取。但是如果,我们的情况是以下情况,又该怎么办? 给所有的类方法添加性能检测,事务控制,该怎么抽取? PerformanceMonitor TransactionManager AOP就是希望将这些分散在各个业务逻辑代码中的相同代码,通过横向切割的方式抽取到一个独立的模块中,让业务逻辑类依然保存最初的单纯。

主要功能

日志记录,性能统计,安全控制,事务处理,异常处理等等。

AOP术语

连接点(Joinpoint)

程序执行的某个特定位置,如某个方法调用前,调用后,方法抛出异常后,这些代码中的特定点称为连接点。简单来说,就是在哪加入你的逻辑增强

切点(PointCut)

每个程序的连接点有多个,如何定位到某个感兴趣的连接点,就需要通过切点来定位。切点用于来限定Spring-AOP启动的范围,通常我们采用表达式的方式来设置,所以关键词是范围

增强(Advice)

增强是织入到目标类连接点上的一段程序代码。在Spring中,像BeforeAdvice等还带有方位信息。

通知(或业务增强)是直译过来的结果, 对照代码就是拦截器定义的相关方法,通知分为如下几种:

前置通知(before):在执行业务代码前做些操作,以在方法中传入JoinPoint对象,用来获得切点信息

后置通知(after):在执行业务代码后做些操作,无论是否发生异常,它都会执行

异常通知(afterThrowing):在执行业务代码后出现异常,需要做的操作,比如回滚事务

返回通知(afterReturning),在执行业务代码后无异常,会执行的操作

环绕通知(around),在建议方法调用之前和之后,执行通知。

目标对象(Target)

需要被加强的业务对象

织入(Weaving)

织入就是将增强添加到对目标类具体连接点上的过程。

代理类(Proxy)

一个类被AOP织入增强后,就产生了一个代理类。

切面(Aspect)

切面由切点和增 zx强组成,它既包括了横切逻辑的定义,也包括了连接点的定义,SpringAOP就是将切面所定义的横切逻辑织入到切面所制定的连接点中。

基于 XML配置

1、引入maven
<!-- 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.aspectj/aspectjweaver -->
<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
     <version>1.9.9.1</version>
</dependency>
2、引入 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:aop="http://www.springframework.org/schema/aop"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:c="http://www.springframework.org/schema/c"
       xmlns:util="http://www.springframework.org/schema/util"

       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/util
        http://www.springframework.org/schema/util/spring-util.xsd"></beans>
3、声明一个切面
<aop:config>
    <aop:aspect id="myAspect" ref="adviceBean">
        ...
    </aop:aspect>
</aop:config>
<bean id="adviceBean" class="...">   <!--adviceBean 既增强  包含要加入到目标类的代码 (可以有多个方法)-->
    ...
</bean>
4、声明一个切入点
<aop:config>
    <aop:aspect id="myAspect" ref="adviceBean">
        <aop:pointcut id="businessService"
                      expression="execution(* com.beiyou.service.*.*(..))"/>
        ...
    </aop:aspect>
</aop:config>
<bean id="adviceBean" class="...">
    ...
</bean>
4.1、切入点表达式
//表达式格式
execution(modifiers? ret-type?  name(param) throws?)
以上所有部分除了ret-type和name以及paramter之外都是可选的returning类型决定了匹配的方法必须要有指定的返回类型,可以使用*表示任意返回类型
只有当方法返回指定类型时,完全限定的类名才会被匹配,name匹配方法名,可以使用*匹配全部或部分方法名
parameters匹配就复杂一下:() 匹配一个无形参的方法,(..)匹配任意数量的形参方法(0~n),(*)匹配带一个任意类型的形参方法
(*,String)匹配带两个形参,第一个任意类型,第二个必须是String类型
​
实例:
execution(public * *(..)):任意public方法
execution(* set*(..)):方法名以set开头的任意方法
execution(* com.xyz.service.AccountService.*(..)):AccountService 接口下的任意方法 
execution(* com.beiyou.service.*.*(..)):service包下的任意类任意方法
重点::execution(* com.xyz.service..*.*(..)):service包或子包下的任意类任意方法
execution(public void MyClass.myMethod(String)) :MyClass 类的myMethod方法,方法public访问权限,void返回值,形参只有一个并为String
​
execution(void MyClass.myMethod(..)):MyClass 类的myMethod方法,任意访问权限,返回值void,任意形参
execution(* MyClass.myMethod(..)):MyClass 类的myMethod方法,任意返回值,任意形参
execution(* MyClass.myMethod*(..)):MyClass 类的以myMethod开头的方法,任意返回值,任意形参
execution(* MyClass.myMethod*(String,..)):MyClass 类的以myMethod开头的方法,任意返回值,第一个形参类型是String
execution(* *.myMethod(..)):任意类下myMethod方法 execution(MyClass.new()):任意MyClass类的无参构造器
execution(MyClass.new(..)):任意MyClass类的任意有参构造器 execution(MyClass+.new(..)):任意MyClass或其子类构造器
execution(public * com.mycompany..*.*(..)):com.mycompany包下任意子包的所有类的所有public 方法
​
//Spring AOP只能切方法(也就是任意连接点中的方法),AspectJ可以切任意成员(任意连接点,包括类/对象初始化块,field,方法,构造器)
within(com.xyz.service.*):service包下任意连接点
within(com.xyz.service..*):service包或子包下任意连接点
this(com.xyz.service.AccountService):AccountService接口的代理实现里的任意连接点
​
target(com.xyz.service.AccountService):目标对象实现了AccountService接口的任意连接点
​
args(java.io.Serializable):只有一个参数且参数在运行时是Serializable类型的任意连接点
@target(org.springframework.transaction.annotation.Transactional):目标对象有一个@Transactional注解任意连接点
@within(org.springframework.transaction.annotation.Transactional):目标对象的声明类型有一个@Transactional注解任意连接点
@annotation(org.springframework.transaction.annotation.Transactional):执行方法有一个@Transactional注解的任意连接点
@args(com.xyz.security.Classified):只有一个参数且参数在运行时参数有@Classified注解的任意连接点
bean(tradeService):Spring bean 名称是tradeService的任意连接点
bean(*Service):Spring bean的名称以Service结尾的任意连接点
//表达式可以使用|| && !进行组合  在XML下就是 or and not
execution(* com.xyz.myapp.service..(..)) and this(service)//xml
​
execution(* com.xyz.myapp.service..(..)) && this(service)//java
5、声明通知
<aop:config>
    <aop:aspect id="myAspect" ref="adviceBean">
        <aop:pointcut id="businessService"
                      expression="execution(* com.by.service.*.*(..))"/>
        <aop:before method="目标方法" pointcut-ref="businessService"/>
        <aop:after method="目标方法" pointcut-ref="businessService"/>
        <aop:after-returning method="目标方法" pointcut-ref="businessService" returning="retVal"/>
        <aop:after-throwing method="目标方法" pointcut-ref="businessService" throwing="ex"/>
        <aop:around method="目标方法" pointcut-ref="businessService"/>
        ...
    </aop:aspect>
</aop:config>
<bean id="adviceBean" class="...">
    ...
</bean>
6、完整代码:
 public static void main( String[] args )
    {
  
        ApplicationContext context=new ClassPathXmlApplicationContext("bean.xml");
        OrderService bean = context.getBean(OrderService.class);
        bean.test();

    }
<?xml version="1.0" encoding="UTF-8"?>

<beans
        xmlns="http://www.springframework.org/schema/beans"
        xmlns:context="http://www.springframework.org/schema/context"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xmlns:p="http://www.springframework.org/schema/p"
        xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd">
  
    <aop:config>
        <aop:aspect id="myAspect" ref="logCut">
            <aop:pointcut id="pointcut1"  expression="execution(* com.beiyou.service.*.*(..))"/>
            <aop:before method="before" pointcut-ref="pointcut1"/>
            <aop:after method="after" pointcut-ref="pointcut1"/>
            <aop:after-returning method="afterReturning" pointcut-ref="pointcut1" returning="retVal"/>
            <aop:after-throwing method="afterThrowing" pointcut-ref="pointcut1" throwing="ex"/>
            <aop:around method="around" pointcut-ref="pointcut1"/>

        </aop:aspect>
    </aop:config>
    <bean id="logCut" class="com.beiyou.LogCut">

    </bean>

    <bean id="orderService" class="com.beiyou.service.OrderService">

    </bean>
</beans>
public class OrderService {

    public  String  test(){

        return "我是orderservice发回值";
    }
}
package com.beiyou;

import org.aspectj.lang.ProceedingJoinPoint;

public class LogCut {

    // 前置通知
    public void before() {
        System.out.println("前置");
    }
    // 后置通知 始终会执行
    public void after() {
        System.out.println("后置");
    }
    // 环绕通知
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("环绕前");
        Object result = pjp.proceed();
        System.out.println("环绕后");
        return result;
    }
    // 后置 正常返回
    public void afterReturning(Object retVal) {
        System.out.println("After returning 后置");
        
    }
    // 发生异常
    public void afterThrowing(Exception ex) {
        System.out.println("发生异常了");
       
    }
}
  public static void main( String[] args )
    {
      
        ApplicationContext context=new ClassPathXmlApplicationContext("bean.xml");
        OrderService bean = context.getBean(OrderService.class);
        bean.test();

    }

基于注解 @AspectJ

1、启用@AsjectJ支持
1.1、xml启用
<?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:p="http://www.springframework.org/schema/p"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:util="http://www.springframework.org/schema/util"
       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 http://www.springframework.org/schema/context/spring-context-4.3.xsd
       http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.3.xsd
       http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop-4.3.xsd"
>
 <!--下面可以不用,通过注解注册也行-->
    <!--下面配置与注解@EnableAspectJAutoProxy 二选一-->
    <aop:aspectj-autoproxy/>    
    <bean id="logCut" class="com.beiyou.LogCut">
    </bean>
    <bean id="orderService" class="com.beiyou.service.OrderService">
    </bean>
</beans>
1.2 注解启用
@Aspect
public class LogCut {

    // 定义切入点
    @Pointcut("execution(* com.beiyou.service.*.*(..))")
    public void point() {}
    // 前置通知
    @Before("point()")
    public void before() {
        System.out.println("前置");
    }
    // 后置通知 始终会执行
    @After("point()")
    public void after() {
        System.out.println("后置");
    }
    // 环绕通知
    @Around("point()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("环绕前");
        Object result = pjp.proceed();
        System.out.println("环绕后");
        return result;
    }
    // 后置 发生异常时不会执行
    @AfterReturning(pointcut="point()",returning="retVal")
    public void afterReturning(Object retVal) {
        System.out.println("返回值是:"+retVal);
        System.out.println("After returning 后置");
    }
    // 发生异常
    @AfterThrowing(pointcut="point()",throwing="ex")
    public void afterThrowing(Exception ex) {
        System.out.println("发生异常了");
    }
}
1.3 完整示例
 public static void main( String[] args )
    {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(Config.class);
       // ApplicationContext context=new ClassPathXmlApplicationContext("bean.xml");
        OrderService bean = context.getBean(OrderService.class);
        bean.test();
      
    }
@ComponentScan
@EnableAspectJAutoProxy
public class Config {
    
}
@Aspect
@Component
public class LogCut {

    // 定义切入点
    @Pointcut("execution(* com.beiyou.service..*.*(..))")
    public void point() {}
    // 前置通知
    @Before("point()")
    public void before() {
        System.out.println("前置");
    }
    // 后置通知 始终会执行
    @After("point()")
    public void after() {
        System.out.println("后置");
    }
    // 环绕通知
    @Around("point()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("环绕前");
        Object result = pjp.proceed();
        System.out.println("环绕后");
        return result;
    }
    // 后置 发生异常时不会执行
    @AfterReturning(pointcut="point()",returning="retVal")
    public void afterReturning(Object retVal) {
        System.out.println("返回值是:"+retVal);
        System.out.println("After returning 后置");
    }
    // 发生异常
    @AfterThrowing(pointcut="point()",throwing="ex")
    public void afterThrowing(Exception ex) {
        System.out.println("发生异常了");
    }
}
@Component
public class OrderService {

    public  String  test(){
        return "我是orderservice返回值";
    }
}
 
  
💡

定义增强类 (相当于切面 ,包含切点和增强)

增强类上增加注解:

@Aspect 说明该类是一个切面

等同于<aop:aspect id="log" ref="logAspect"></aop:aspect>

@Component 将增强类注入到ioc容器中

@EnableAspectJAutoProxy 启用注解支持

等同于 <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

这孩子叫逆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值