6. Spring的AOP

一、基本概念

1. AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善
2. 利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面
3. 所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性
4. 使用"横切"技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点
5. 横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事务。AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来

二、核心名词

1. 横切关注点题
1) 对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
2. 切面(aspect)
1) 类是对物体特征的抽象,切面就是对横切关注点的抽象

2) 是切入点和通知的结合,以后咱们自己来编写和配置的
3. 连接点(joinpoint)
1) 被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器

2) 程序运行中的一些时间点, 例如一个方法的执行, 或者是一个异常的处理

4. 切入点(pointcut)
1) 对连接点进行拦截的定义

2) 在 Spring 中, 所有的方法都可以认为是 joinpoint, 但是我们并不希望在所有的方法上都添加 Advice, 而 pointcut 的作用就是提供一组规则(使用 AspectJ pointcut expression language 来描述) 来匹配joinpoint, 给满足规则的 joinpoint 添加 Advice

3) advice 是在 join point 上执行的, 而 point cut 规定了哪些 join point 可以执行哪些 advice
5. 通知(advice)
1) 所谓通知指的就是指拦截到连接点之后要执行的代码

2) 通知种类:

	(1)前置通知: 在目标类的方法执行之前执行

	 	配置文件信息:<aop:after method="before" pointcut-ref="myPointcut3"/>

	 	应用:可以对方法的参数来做校验

	(2)最终通知: 在目标类的方法执行之后执行,如果程序出现了异常,最终通知也会执行	

		配置文件信息:<aop:after method="after" pointcut-ref="myPointcut3"/>

		应用:例如像释放资源

	(3)后置通知:	方法正常执行后的通知	

		配置文件信息:<aop:after-returning method="afterReturning" pointcut-ref="myPointcut2"/>

		应用:可以修改方法的返回值

	(4)异常抛出通知:在抛出异常后通知	

		配置文件信息:<aop:after-throwing method="afterThorwing" pointcut-ref="myPointcut3"/>

		应用:包装异常的信息

	(5)环绕通知:方法的执行前后执行	

		配置文件信息: <aop:around method="around" pointcut-ref="myPointcut2"/>

		要注意:目标的方法默认不执行,需要使用ProceedingJoinPoint对来让目标对象的方法执行

6. 目标对象
1) 代理的目标对象

2) 织入 advice 的目标对象. 目标对象也被称为 advised object
7. 织入(weave)
1) 将切面应用到目标对象并导致代理对象创建的过程

2) 将 aspect 和其他对象连接起来, 并创建 adviced object 的过程
8.引入(introduction)
1) 在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
9. Proxy(代理)
1) 一个类被AOP织入增强后,就产生一个结果代理类

2) 它是融合了原类和增强逻辑的代理类.在 Spring AOP 中, 一个 AOP 代理是一个 JDK 动态代理对象或 CGLIB 代理对象.

三、AOP编程

1. Spring对AOP的支持
Spring中AOP代理由Spring的IOC容器负责生成、管理,其依赖关系也由IOC容器负责管理。因此,AOP代理可以直接使用容器中的其它bean实例作为目标,这种关系可由IOC容器的依赖注入提供。Spring创建代理的规则为:

1)默认使用Java动态代理来创建AOP代理,这样就可以为任何接口实例创建代理了

2)当需要代理的类不是代理接口的时候,Spring会切换为使用CGLIB代理,也可强制使用CGLIB

2. aop编程步骤步骤:进行AOP编程的关键就是定义切入点和定义增强处理,一旦定义了合适的切入点和增强处理,AOP框架将自动生成AOP代理 (代理对象的方法=增强处理+被代理对象的方法)
1)定义普通业务组件

2)定义切入点,一个切入点可能横切多个业务组件

3)定义增强处理,增强处理就是在AOP框架为普通业务组件织入的处理动作
3. springAOP的具体加载步骤
1)当 spring 容器启动的时候,加载了 spring 的配置文件

2)为配置文件中的所有 bean 创建对象

3) spring 容器会解析 aop:config 的配置

4) 解析切入点表达式,用切入点表达式和纳入spring容器中的bean做匹配, 如果匹配成功,则会为该bean创建代理对象,代理对象的方法=目标方法+通知,如果匹配不成功,不会创建代理对象

5) 在客户端利用 context.getBean() 获取对象时,如果该对象有代理对象,则返回代理对象;如果没有,则返回目标对象

4. 一般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:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-4.2.xsd">
        
        <bean id="helloWorldImpl1" class="com.xrq.aop.HelloWorldImpl1" />
        <bean id="helloWorldImpl2" class="com.xrq.aop.HelloWorldImpl2" />
        <bean id="timeHandler" class="com.xrq.aop.TimeHandler" />
        <bean id="logHandler" class="com.xrq.aop.LogHandler" />
        
        <aop:config>
            <aop:aspect id="time" ref="timeHandler" order="1">
                <aop:pointcut id="addTime" expression="execution(* com.xrq.aop.HelloWorld.print*(..))" />
                <aop:before method="printTime" pointcut-ref="addTime" />
                <aop:after method="printTime" pointcut-ref="addTime" />
            </aop:aspect>
            <aop:aspect id="log" ref="logHandler" order="2">
                <aop:pointcut id="printLog" expression="execution(* com.xrq.aop.HelloWorld.do*(..))" />
                <aop:before method="LogBefore" pointcut-ref="printLog" />
                <aop:after method="LogAfter" pointcut-ref="printLog" />
            </aop:aspect>
        </aop:config>
</beans>

5. 注解配置
1)写切面类

	@Aspect
	@Component
	public class ReadThroughAssignAop extends SingleReadCacheAdvice {

	    private static final Logger LOG = LoggerFactory.getLogger(ReadThroughAssignAop.class);

	    // 定义切入点,定义切入表达式(带@ReadCache注解的)
	    @Pointcut("@annotation(com.pagoda.redis.annotation.ReadCache)")
	    public void getSingleAssign() {

	    }

	    // 定义通知,以及需要连接到的切入点(通过方法名关联上)
	    @Around("getSingleAssign()")
	    public Object cacheSingleAssign(final ProceedingJoinPoint jp) throws Throwable {
	        // to do 作通知的操作
	        return cache(jp);
	    }

	    @Override
	    protected Logger getLogger() {
	        return LOG;
	    }


	}

2)spring中配置

    <!-- redis缓存运行切面 -->
    <bean id="RedisCacheAspect"  class="com.pagoda.redis.aop.ReadThroughAssignAop" />

    <!-- 切面申明配置-->
    <aop:aspectj-autoproxy>
        <aop:include name="RedisCacheAspect" />
    </aop:aspectj-autoproxy>	

四、运用场景
1. 性能检测
2. 权限验证
3. 日志记录
4. 事务控制
5. 集成redis 
6. 错误处理
7. 缓存

参考网址

Spring3:AOP

彻底征服 Spring AOP 之 理论篇

彻底征服 Spring AOP 之 实战篇

Spring思维导图,让Spring不再难懂(aop篇)

Spring框架的核心功能之AOP技术

注:文章是经过参考其他的文章然后自己整理出来的,有可能是小部分参考,也有可能是大部分参考,但绝对不是直接转载,觉得侵权了我会删,我只是把这个用于自己的笔记,顺便整理下知识的同时,能帮到一部分人。
ps : 有错误的还望各位大佬指正,小弟不胜感激

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
java.lang.NullPointerException: null at com.datech.web.controller.system.TbappController.remove(TbappController.java:667) at com.datech.web.controller.system.TbappController$$FastClassBySpringCGLIB$$ea1c3ba.invoke(<generated>) at org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:218) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.invokeJoinpoint(CglibAopProxy.java:771) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:163) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:749) at org.springframework.aop.aspectj.AspectJAfterThrowingAdvice.invoke(AspectJAfterThrowingAdvice.java:62) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:175) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:749) at org.springframework.aop.framework.adapter.AfterReturningAdviceInterceptor.invoke(AfterReturningAdviceInterceptor.java:55) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:175) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:749) at org.apache.shiro.spring.security.interceptor.AopAllianceAnnotationsAuthorizingMethodInterceptor$1.proceed(AopAllianceAnnotationsAuthorizingMethodInterceptor.java:82) at org.apache.shiro.authz.aop.AuthorizingMethodInterceptor.invoke(AuthorizingMethodInterceptor.java:39) at org.apache.shiro.spring.security.interceptor.AopAllianceAnnotationsAuthorizingMethodInterceptor.invoke(AopAllianceAnnotationsAuthorizingMethodInterceptor.java:115) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:749) at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:95) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186) at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:749) at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:691) at com.datech.web.controller.system.TbappController$$EnhancerBySpringCGLIB$$b3b02033.remove(<generated>) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
最新发布
07-14

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值