博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring核心系列之AOP(一)
阅读量:6000 次
发布时间:2019-06-20

本文共 5882 字,大约阅读时间需要 19 分钟。

Spring核心系列之AOP(一)

Hello,大家好,今天来给大家讲一讲Spring中的AOP,面向切面编程,它在Spring的整个体系中占有着重要地位。本文还是以实践为主,注解切入注入,OK,文章结构:

  1. @AspectJ 详解
  2. Spring AOP - AspectJ注解

1. @AspectJ 的由来

提到AspectJ,其实很多人是有误解的,很多人只知道在Spring中使用Aspect那一套注解,以为是Spring开发的这一套注解,这里我觉得有责任和大家澄清一下。 AspectJ是一个AOP框架,它能够对java代码进行AOP编译(一般在编译期进行),让java代码具有AspectJ的AOP功能(当然需要特殊的编译器),可以这样说AspectJ是目前实现AOP框架中最成熟,功能最丰富的语言,更幸运的是,AspectJ与java程序完全兼容,几乎是无缝关联,因此对于有java编程基础的工程师,上手和使用都非常容易. 其实AspectJ单独就是一门语言,它需要专门的编译器(ajc编译器). Spring AOP 与ApectJ的目的一致,都是为了统一处理横切业务,但与AspectJ不同的是,Spring AOP并不尝试提供完整的AOP功能(即使它完全可以实现),Spring AOP 更注重的是与Spring IOC容器的结合,并结合该优势来解决横切业务的问题,因此在AOP的功能完善方面,相对来说AspectJ具有更大的优势,同时,Spring注意到AspectJ在AOP的实现方式上依赖于特殊编译器(ajc编译器),因此Spring很机智回避了这点,转向采用动态代理技术的实现原理来构建Spring AOP的内部机制(动态织入),这是与AspectJ(静态织入)最根本的区别。在AspectJ 1.5后,引入@Aspect形式的注解风格的开发,Spring也非常快地跟进了这种方式,因此Spring 2.0后便使用了与AspectJ一样的注解。请注意,Spring 只是使用了与 AspectJ 5 一样的注解,但仍然没有使用 AspectJ 的编译器,底层依是动态代理技术的实现,因此并不依赖于 AspectJ 的编译器。 所以,大家要明白,Spring AOP虽然是使用了那一套注解,其实实现AOP的底层是使用了动态代理(JDK或者CGLib)来动态植入。至于AspectJ的静态植入,不是本文重点,所以只提一提。

2. Spring AOP - AspectJ注解

谈到Spring AOP,老程序员应该比较清楚,之前的Spring AOP没有使用@Aspect这一套注解和aop:config这一套XML解决方案,而是开发者自己定义一些类实现一些接口,而且配置贼恶心。本文就不再演示了,后来Spring痛下决心,把AspectJ"整合"进了Spring当中,并开启了aop命名空间。现在的Sping AOP可以算是朗朗乾坤。上例子之前,还是把AOP的概念都提一提:

  • 切点:定位到具体方法的一个表达式
  • 切面: 切点+建言
  • 建言(增强):定位到方法后干什么事

网上有很多概念,什么连接点,织入,目标对象,引入什么的。我个人觉得,在Java的Spring AOP领域,完全不用管。别把自己绕晕了。就按照我说的这三个概念就完事了,好了,先来搞个例子: maven依赖:

org.springframework
spring-core
4.2.3.RELEASE
org.springframework
spring-context
4.2.3.RELEASE
org.springframework
spring-aop
4.2.3.RELEASE
//下面这两个aspectj的依赖是为了引入AspectJ的注解
org.aspectj
aspectjrt
1.6.12
org.aspectj
aspectjweaver
1.6.12
//Spring AOP底层会使用CGLib来做动态代理
cglib
cglib
2.2
复制代码

小狗类,会说话:

public class Dog {    private String name;    public void say(){        System.out.println(name + "在汪汪叫!...");    }    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }}复制代码

切面类:

@Aspect //声明自己是一个切面类public class MyAspect {    /**     * 前置通知     */     //@Before是增强中的方位     // @Before括号中的就是切入点了     //before()就是传说的增强(建言):说白了,就是要干啥事.    @Before("execution(* com.zdy..*(..))")    public void before(){        System.out.println("前置通知....");    }}复制代码

这个类是重点,先用@Aspect声明自己是切面类,然后before()为增强,@Before(方位)+切入点可以具体定位到具体某个类的某个方法的方位. Spring配置文件:

//开启AspectJ功能.    
复制代码

然后Main方法:

ApplicationContext ac =new ClassPathXmlApplicationContext("applicationContext.xml");        Dog dog =(Dog) ac.getBean("dog");        System.out.println(dog.getClass());        dog.say();复制代码

输出结果:

class com.zdy.Dog$$EnhancerBySpringCGLIB$$80a9ee5f前置通知....null在汪汪叫!...复制代码

说白了,就是把切面类丢到容器,开启一个AdpectJ的功能,Spring AOP就会根据切面类中的(@Before+切入点)定位好具体的类的某个方法(我这里定义的是com.zdy包下的所有类的所有方法),然后把增强before()切入进去.

然后说下Spring AOP支持的几种类似于@Before的AspectJ注解:

  • 前置通知@Before: 前置通知通过@Before注解进行标注,并可直接传入切点表达式的值,该通知在目标函数执行前执行,注意JoinPoint,是Spring提供的静态变量,通过joinPoint 参数,可以获取目标对象的信息,如类名称,方法参数,方法名称等,该参数是可选的。
@Before("execution(...)")public void before(JoinPoint joinPoint){    System.out.println("...");}复制代码
  • 后置通知@AfterReturning: 通过@AfterReturning注解进行标注,该函数在目标函数执行完成后执行,并可以获取到目标函数最终的返回值returnVal,当目标函数没有返回值时,returnVal将返回null,必须通过returning = “returnVal”注明参数的名称而且必须与通知函数的参数名称相同。请注意,在任何通知中这些参数都是可选的,需要使用时直接填写即可,不需要使用时,可以完成不用声明出来。
@AfterReturning(value="execution(...)",returning = "returnVal")public void AfterReturning(JoinPoint joinPoint,Object returnVal){   System.out.println("我是后置通知...returnVal+"+returnVal);}复制代码
  • 异常通知 @AfterThrowing:该通知只有在异常时才会被触发,并由throwing来声明一个接收异常信息的变量,同样异常通知也用于Joinpoint参数,需要时加上即可.
@AfterThrowing(value="execution(....)",throwing = "e")public void afterThrowable(Throwable e){  System.out.println("出现异常:msg="+e.getMessage());}复制代码
  • 最终通知 @After:该通知有点类似于finally代码块,只要应用了无论什么情况下都会执行.
@After("execution(...)")public void after(JoinPoint joinPoint) {    System.out.println("最终通知....");}复制代码
  • 环绕通知 @Around: 环绕通知既可以在目标方法前执行也可在目标方法之后执行,更重要的是环绕通知可以控制目标方法是否指向执行,但即使如此,我们应该尽量以最简单的方式满足需求,在仅需在目标方法前执行时,应该采用前置通知而非环绕通知。案例代码如下第一个参数必须是ProceedingJoinPoint,通过该对象的proceed()方法来执行目标函数,proceed()的返回值就是环绕通知的返回值。同样的,ProceedingJoinPoint对象也是可以获取目标对象的信息,如类名称,方法参数,方法名称等等
@Around("execution(...)")public Object around(ProceedingJoinPoint joinPoint) throws Throwable {    System.out.println("我是环绕通知前....");    //执行目标函数    Object obj= (Object) joinPoint.proceed();    System.out.println("我是环绕通知后....");    return obj;}复制代码

然后说下一直用"..."忽略掉的切入点表达式,这个表达式可以不是exection(..),还有其他的一些,我就不说了,说最常用的execution:

//scope :方法作用域,如public,private,protect//returnt-type:方法返回值类型//fully-qualified-class-name:方法所在类的完全限定名称//parameters 方法参数execution(
.*(parameters))复制代码
.*(parameters)复制代码

注意这一块,如果没有精确到class-name,而是到包名就停止了,要用两个".."来表示包下的任意类:

  • execution(* com.zdy..*(..)):com.zdy包下所有类的所有方法.
  • execution(* com.zdy.Dog.*(..)): Dog类下的所有方法.

具体详细语法,大家如果有需求自行google了,我最常用的就是这俩了。要么按照包来定位,要么按照具体类来定位.

在使用切入点时,还可以抽出来一个@Pointcut来供使用:

/** * 使用Pointcut定义切点 */@Pointcut("execution(...)")private void myPointcut(){}/** * 应用切入点函数 */@After(value="myPointcut()")public void afterDemo(){    System.out.println("最终通知....");}复制代码

可以避免重复的execution在不同的注解里写很多遍...

结语

好了,Spring AOP 基于AspectJ注解如何实现AOP给大家分享完了,其实不是很难,因为博主比较懒,主说了主要内容,像切入点表达式要说其实有很多东西可以说,还有增强级别的排序问题,比如一个Aspect类中定义了多个@Before,谁先调用,因为博主觉得这些知识点不是很重要,用的非常少,所以就没提了。这一期把基于注解的分享完了,下一次准备把Spring AOP基于XML的配置讲一讲,然后说一些实际的运用场景。包括多个切面命中一个切点时,切面级别的调用顺序等都会提一提。然后把Spring AOP底层使用的JDK动态代理和CGLib动态代理都稍微说一说。好了,这一期就到这里了..Over,Have a good day .

转载地址:http://hozmx.baihongyu.com/

你可能感兴趣的文章
Eclipse下 java.lang.OutOfMemoryError: PermGen space
查看>>
我的友情链接
查看>>
华为和思科的默认路由优先级及路由选择
查看>>
内外网通过公网IP访问DMZ主机
查看>>
5、用户和用户组管理
查看>>
报错:cannot execute binary file
查看>>
Artemis架构解析
查看>>
checkpoint 防火墙系统报错1--the internal CA certificate failed(内部CA损坏)
查看>>
1级增量备份脚本(示例)
查看>>
西安OpenParty11月29日活动高清图文回顾——新增西安APEC蓝美图!
查看>>
如何提高PHP应用的性能
查看>>
打破陈规抓痛点,H3 BPM10.0挑战不可能
查看>>
unit5.6
查看>>
Spring Cloud+Docker创建微服务容器实例
查看>>
Android多进程介绍
查看>>
win7自带FTP
查看>>
谷歌释放改造蚊子 会带来意外的灾难性后果吗?
查看>>
ZooKeeper源码研究系列(1)源码环境搭建
查看>>
高端大气上档次·玩转微信摇色子
查看>>
【小松教你手游开发】【系统模块开发】图文混排 (在label中插入表情)
查看>>