Spring AOP(完结)

一、第五种方式

1、AOP的相关概念

  • 横切关注点:对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
  • Aspect(切面):通常是一个类,里面可以定义切入点和通知
  • JointPoint(连接点):程序执行过程中明确的点,一般是方法的调用。被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
  • Advice(通知):AOP在特定的切入点上执行的增强处理,有before(前置),after(后置),afterReturning(最终),afterThrowing(异常),around(环绕)
  • Pointcut(切入点):就是带有通知的连接点,在程序中主要体现为书写切入点表达式
  • weave(织入):将切面应用到目标对象并导致代理对象创建的过程
  • introduction(引入):在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
  • AOP代理(AOP Proxy):AOP框架创建的对象,代理就是目标对象的加强。Spring中的AOP代理可以使JDK动态代理,也可以是CGLIB代理,前者基于接口,后者基于子类
  • 目标对象(Target Object): 包含连接点的对象。也被称作被通知或被代理对象。POJO

2、第五种方式

Student.class

package com.qianfeng.aop05;


public class Student implements Person {
    @Override
    public boolean eat() {
        System.out.println("I can eat");
        return true;
    }
    @Override
    public String drink() {
        System.out.println(1/0);
        System.out.println("I can drink");
        return null;
    }
}

beans5.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: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/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd">
    <bean id="st" class="com.qianfeng.aop05.Student" />
    <bean id="my" class="com.qianfeng.aop05.MyAspect" />
    <aop:config proxy-target-class="true">
        <aop:aspect ref="my">
            <aop:pointcut id="pt" expression="(execution(boolean com.qianfeng.aop05.*.*(..))) or (execution(java.lang.String com.qianfeng.aop05.*.*(..)))"/>
            <aop:around method="myAround" pointcut-ref="pt"/>
            <aop:before method="myBefore" pointcut-ref="pt"/>
            <aop:after method="myAfter" pointcut-ref="pt"/>

            <aop:after-returning method="myReturn" pointcut-ref="pt" returning="object"/>
            <aop:after-throwing method="myThrow" pointcut-ref="pt" throwing="e"/>
        </aop:aspect>
    </aop:config>
</beans>
  • aop:point-cut标签指定了切点,返回值、包、类、方法与参数,只匹配符合条件的方法,也就是对这些方法进行代理,条件可以进行与或非操作,分别使用and或&&、or或||、!进行多条件匹配。
  • aop:before标签是前置通知,在切点方法执行前通知
  • aop:around标签是环绕通知,在方法执行前后都通知
  • aop:after标签是后置通知,在方法执行后通知
  • aop:after-returning标签是后置返回通知,方法执行后的返回值进行通知,returning参数所对应的值是方法返回的参数用于放在method方法中的形参。
  • aop:after-throwing是异常抛出通知,只有方法的执行过程中出现了异常并且自己没有处理的话才会执行通知,throwing是捕获到的异常用于放在对应method方法中的形参
  • 执行顺序:before >around before > 业务方法 > (after-throwing)(after returning > around after)>after。不过顺序也不固定:使用注解的话是环绕通知proceed方法之前部分先执行,使用xml配置的话取决于aop:before和aop:around的配置顺序

MyAspect.java

package com.qianfeng.aop05;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;

import java.util.Arrays;

public class MyAspect{
    public void myBefore(JoinPoint jp){
        System.out.println("args:"+ Arrays.toString(jp.getArgs()));
        System.out.println("toString:"+jp.toString());
        System.out.println("getTarget:"+jp.getTarget());
        System.out.println("---------before----------");
    }
    public void myAfter(){
        System.out.println("---------after----------");
    }
    public Object myAround(ProceedingJoinPoint pjp){
        System.out.println("this is around-before");
        try {
            Object obj = pjp.proceed();
            System.out.println("this is around-after"+obj);
            return obj;
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        return null;
    }

    public void myReturn(JoinPoint jp,Object object){
        System.out.println("this is after returning "+object);
    }

    public void myThrow(JoinPoint jp,Throwable e){
        System.out.println("this is after throwing "+e.getMessage());
    }
}

  • JoinPoint是连接点,会包含连接点的基本消息,通知所在的类、通知的方法名、通知方法的入参集合。getArgs()方法获取方法的参数数组,getTarget()获取目标对象信息
  • ProceedingJoinPoint是JoinPoint的子接口,该对象只用在Around的切面方法中。

二、第六种方式

第五种方式使用xml文件配置较多步骤,比较麻烦,第六种方式使用注解更为简便

beans6.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:aop="http://www.springframework.org/schema/aop"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">
    <context:component-scan base-package="com.qianfeng.aop06"/>
    <aop:aspectj-autoproxy/>
</beans>

相较于第五种方式,第六种方式主要的不同就是可以实现自动代理,xml文件的配置有以下不同:

  • 引入了三个命名空间,和context有关
  • 添加了context:component-scan标签,用于扫描组件,base-package指定了需要进行组件扫描的包
  • aop:aspectj-autoproxy标签指定了实现自动代理

MyAspect.java

package com.qianfeng.aop06;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

import java.util.Arrays;
@Component
@Aspect
public class MyAspect{
    @Pointcut(value = "execution(boolean com.qianfeng.aop06.*.*(..))")
    public void setAll(){}
    @Before(value = "execution(java.lang.String com.qianfeng.aop06.*.*(..))")
    public void myBefore(JoinPoint jp){
        System.out.println("args:"+ Arrays.toString(jp.getArgs()));
        System.out.println("toString:"+jp.toString());
        System.out.println("getTarget:"+jp.getTarget());
        System.out.println("---------before----------");
    }
    @After("setAll()")
    public void myAfter(){
        System.out.println("---------after----------");
    }
    @Around("setAll()")
    public Object myAround(ProceedingJoinPoint pjp){
        System.out.println("this is around-before");
        try {
            Object obj = pjp.proceed();
            System.out.println("this is around-after"+obj);
            return obj;
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        return null;
    }
    @AfterReturning(value = "setAll()",returning = "object")
    public void myReturn(JoinPoint jp,Object object){
        System.out.println("this is after returning "+object);
    }
    @AfterThrowing(value = "setAll()",throwing = "e")
    public void myThrow(JoinPoint jp,Throwable e){
        System.out.println("this is after throwing "+e.getMessage());
    }
}

  • @Component注解标注当前类是一个组件,在扫描时会被扫描进来
  • @Aspect标注当前类是一个切面类
  • @PointCut标注切点,为了避免相同的匹配规则被定义多处,专门定义该方法设置执行的匹配规则,各个方法自行调用即可
  • @Before标注前置通知
  • @After标注后置通知
  • @Around标注环绕通知
  • @AfterReturning标注后置返回通知
  • @AfterThorwing标注抛出异常通知
  • 注解的执行顺序稍有不同:(after throwing)around before>before > 业务方法 > around-after > after >after returning

三、第七种方式

第七种方式使用BeanPostProcessor方式实现。该接口我们也叫后置处理器,作用是在Bean对象在实例化和依赖注入完毕后,在显示调用初始化方法的前后添加我们自己的逻辑。注意是Bean实例化完毕后及依赖注入完成后触发的。

Spring中Bean的运行顺序:

  • Spring IoC容器实例化Bean(注意实例化与初始化的区别,实例化是在内存中开辟空间,初始化是对变量赋值)
  • 调用BeanPostProcesson的postProcessBeforeInitialization方法
  • 调用Bean实例的初始化方法
  • 调用BeanPostProcesson的postProcessAfterInitialization方法

beans7.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:aop="http://www.springframework.org/schema/aop"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">
    <context:component-scan base-package="com.qianfeng.aop07"/>
    <bean class="com.qianfeng.aop07.MyBeanPostProcessor"/>
</beans>
  • context:component-scan用于指定组件扫描,base-package指定需要扫描的包,只有打上@component注解的类才会被扫描
  • bean class="com.qianfeng.aop07.MyBeanPostProcessor"指定BeanPostProcessor的Factory hook,让每个bean对象初始化是自动回调该对象中的回调方法

MyBeanPostProcessor.java

package com.qianfeng.aop07;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class MyBeanPostProcessor implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("this is before "+bean);
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("this is after");
        return Proxy.newProxyInstance(MyBeanPostProcessor.class.getClassLoader(), bean.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

                MyAspect ma = new MyAspect();

                ma.before();

                Object obj = method.invoke(bean, args);

                ma.after();

                return obj;
            }
        });

    }
}

  • 这个类实现了BeanPostProcessor接口,重写了postProcessBeforeInitialization()与postProcessAfterInitialization()方法,这两个方法的第一个参数是目标对象的bean,返回值都是Object类型,因为把bean从IoC容器中取出来还需要放回去,如果返回null的话会报异常
  • postProcessBeforeInitialization()是在初始化方法之前执行,postProcessAfterInitialization()是在初始化方法之后执行
  • 配置了包扫描之后,该类会初始化两个对象EventListenerMethodProcessor和DefaultEventListenerFactory,再外加我们自己的组件对象 , 所以会发现有三个before打印。
  • return Proxy.newProxyInstance(MyBeanPostProcessor.class.getClassLoader(), bean.getClass().getInterfaces(), new InvocationHandler() ...)这条语句的作用就是将代理后的对象放进IoC容器中。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容