10 Spring框架 AOP (三) Spring对AspectJ的整合
上两节我们讲了Spring对AOP的实现,但是在我们的开发中我们不太使用Spring自身的对AOP的实现,而是使用AspectJ,AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法,所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件,如果我们要单独的使用Aspect,我们需要安装,并配置环境变量,但是Spring对AspectJ做了很好的整合,我们只需要将相关的jar包导入,就可以在Spring的环境下使用AspectJ。
本节主要讲解Spring环境下Aspect的环境搭建,和在Spring环境下使用AspectJ的AOP,大体的内容摘要如下所示:
- AspectJ环境搭建
- Aspect 注解方式的各种通知实现
- Aspect xml方法的各种通知实现
(一)AspectJ环境搭建
环境搭建分为两步:
①导jar包
我们本节AspectJ的环境搭建是在Spring4.X的基础上实现的,所以首先必须的是SpringAOP环境的搭建,我们这里需要使用到五个Jar包(我们所涂鸦的五个):
第一个是AOP联盟的jar包,第二个,第三个是AspectJ的jar包,第四个是SpringAOP的jar包,最后一个是Spring整合AspectJ的jar包。
②改变约束文件
我们应该在
spring-framework-4.1.6.RELEASE-dist/spring-framework-4.1.6.RELEASE/docs/spring-framework-reference/html/xsd-config.html
这个目录下找到约束文件:
这样我们的环境就搭建好了。
(二)Aspect 注解方式的各种通知实现
好多步骤解释我都写在了注解中,代码就不做详细的介绍了。
首先我们给出我们的基础代码(测试使用):
//一个接口
public interface SomeServices {
void doFirst();
String doSecond();
void doThird();
}
//一个目标类(需要加强方法类)
public class SomeServiceImp implements SomeServices{
@Override
public void doFirst() {
System.out.println("print first");
}
@Override
//这个方法有返回值,供后置通知和环绕通知测试
public String doSecond() {
System.out.println("print second");
return "abc";
}
@Override
public void doThird() {
//System.out.println("print third");
//使得程序产生异常,测试异常通知
System.out.println("print third" + 3/0);
}
}
<!--AspectJ实现AOP的配置文件-->
<?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="service" class="com.test.basedAnnotion.SomeServiceImp"/>
<bean id="myAspect" class="com.test.basedAnnotion.MyAspect"/>
<!-- 自动AspectJ代理 -->
<aop:aspectj-autoproxy/>
</beans>
在这个配置文件中,我们要注册需要加强的目标类bean,和我们的切面bean,还有,因为我们是基于注解的,所以我们需要像IOC注解使用那样开启扫描,开启AspectJ的自动代理,扫描我们的组件,并自动为我们产生代理。
下面我们的主角就要上场了,大家也应该猜到了,就是我们定义的切面类(代码比较庞大,由所有通知组成):
@Aspect
public class MyAspect {
//前置通知
@Before("execution(* *..SomeServices.doFirst(..))")
public void before() {
System.out.println("前置通知");
}
//前置通知
@Before("execution(* *..SomeServices.doFirst(..))")
public void before(JoinPoint jp) {//这个参数可以获取切入点路径
System.out.println("jp:"+jp);
}
//后置通知
@AfterReturning("execution(* *..SomeServices.doSecond(..))")
public void AfterReturning() {
System.out.println("后置通知");
}
//后置通知,和SpringAOP一样,我们可以获得主业务方法的返回值
@AfterReturning(value="execution(* *..SomeServices.doSecond(..))",returning="result")
public void AfterReturning(String result) {
System.out.println("后置通知");
System.out.println("result=" + result);
}
//环绕通知
@Around(value="execution(* *..SomeServices.doSecond(..))")
public Object around(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("环绕执行前置方法");
//这里我们可以对主业务方法的返回值进行修改
String result = ((String)pjp.proceed()).toUpperCase();
System.out.println("环绕执行后置方法");
return result;
}
//异常通知
@AfterThrowing("execution(* *..SomeServices.doThird(..))")
public void afterThrowing() {
System.out.println("异常通知");
}
//异常通知,可以获取异常参数
@AfterThrowing(value="execution(* *..SomeServices.doThird(..))",throwing="e")
public void afterThrowing(Exception e) {
System.out.println("异常通知" + e.getMessage());
}
//最终通知
@After("execution(* *..SomeServices.doThird(..))")
public void after() {
System.out.println("最终通知");
}
//如果为一个方法定义多个通知,那么我们就需要写很多切入点,所以我们可以写一个通用的切入点,供上面的通知使用
@Pointcut("execution(* *..SomeServices.doThird(..))")
public void doThirdPointCut() {}
//下面我们就来使用这个定义的切入点
//使用定义切入点的最终通知
@After("doThirdPointCut()")
public void after2() {
System.out.println("定义切入点的最终通知");
}
}
首先我们看到的是Aspect注解,这个注解表明当前的类是一个切面,就像IOC中的几个注解一样(Component,Service...),表明当前类的作用和地位。
上面的几种通知注解(Before,AfterReturning...)后面跟的参数是切入点表达式,关于切入点的相关介绍本节仅做简单的介绍。
切入点:通俗的讲切入点就是我们切面添加的位置
切入点表达式:它由“*,空格,..相关的方法名,包名”组成,
大概的格式为:方法返回值类型+包名+类名+方法名+方法参数。
最后我们来进行测试:
public class test {
@Test
public void Test01() {
String source = "com/test/basedAnnotion/applicationContext.xml";
ApplicationContext ac = new ClassPathXmlApplicationContext(source);
SomeServices service = (SomeServices)ac.getBean("service");
service.doFirst();
System.out.println("-----------");
String result = service.doSecond();
System.out.println(result);
System.out.println("-----------");
service.doThird();
}
}
控制台输出:
jp:execution(void com.test.basedAnnotion.SomeServices.doFirst())
前置通知
print first
-----------
环绕执行前置方法
print second
环绕执行后置方法
后置通知
result=ABC
后置通知
ABC
-----------
最终通知
定义切入点的最终通知
异常通知/ by zero
异常通知
(三)Aspect xml方法的各种通知实现
基本的代码(一个接口,一个实现类,一个切面类,一个测试类),都和上面的相同,当然切面类中可以将注解删除。
下面就是我们的xml配置文件:
<!-- 注册目标类 -->
<bean id="service" class="com.test.basedXml.SomeServiceImp"/>
<!-- 注册切面 -->
<bean id="myAspect" class="com.test.basedXml.MyAspect"/>
<!-- AOP配置 -->
<aop:config>
<!-- 就像我们注解方式一样,我们也可以定义切入点,能够复用 -->
<aop:pointcut expression="execution(* *..SomeServices.doFirst(..))" id="doFirst"/>
<aop:pointcut expression="execution(* *..SomeServices.doSecond(..))" id="doSecond"/>
<aop:pointcut expression="execution(* *..SomeServices.doThird(..))" id="doThird"/>
<aop:aspect ref="myAspect">
<!-- 前置通知 -->
<!-- 直接指定pointcut -->
<aop:before method="before" pointcut="execution(* *..SomeServices.doFirst(..))"/>
<!--ref指定pointcut -->
<!-- <aop:before method="before" pointcut-ref="doFirst"/> -->
<!-- 指定带参数的通知方法 -->
<aop:before method="before(org.aspectj.lang.JoinPoint)" pointcut-ref="doFirst"/>
<!-- 后置通知 -->
<aop:after-returning method="AfterReturning" pointcut-ref="doSecond"/>
<!-- 如果后置通知想要获取返回值,要加上参数returning -->
<aop:after-returning method="AfterReturning(java.lang.String)" pointcut-ref="doSecond" returning="result"/>
<!-- 环绕通知 -->
<aop:around method="around" pointcut-ref="doSecond" />
<!-- 异常通知 -->
<aop:after-throwing method="afterThrowing" pointcut-ref="doThird"/>
<!-- 异常返回值需要加上throwing参数 -->
<aop:after-throwing method="afterThrowing(java.lang.Exception)" pointcut-ref="doThird" throwing="e"/>
<!-- 最终通知 -->
<aop:after method="after" pointcut-ref="doThird"/>
</aop:aspect>
</aop:config>
以后我们使用Spring整合的AspectJ都使用这种xml形式的。
本章代码均经过博主测试,以供博主以后查阅,也能够供大家参考,如有错误欢迎指正!
10 Spring框架 AOP (三) Spring对AspectJ的整合的更多相关文章
- spring框架 AOP核心详解
AOP称为面向切面编程,在程序开发中主要用来解决一些系统层面上的问题,比如日志,事务,权限等待,Struts2的拦截器设计就是基于AOP的思想,是个比较经典的例子. 一 AOP的基本概念 (1)Asp ...
- 跟着刚哥学习Spring框架--AOP(五)
AOP AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善.OOP引入 ...
- Spring框架系列(9) - Spring AOP实现原理详解之AOP切面的实现
前文,我们分析了Spring IOC的初始化过程和Bean的生命周期等,而Spring AOP也是基于IOC的Bean加载来实现的.本文主要介绍Spring AOP原理解析的切面实现过程(将切面类的所 ...
- Spring框架系列(12) - Spring AOP实现原理详解之JDK代理实现
上文我们学习了SpringAOP Cglib动态代理的实现,本文主要是SpringAOP JDK动态代理的案例和实现部分.@pdai Spring框架系列(12) - Spring AOP实现原理详解 ...
- spring框架aop用注解形式注入Aspect切面无效的问题解决
由于到最后我的项目还是有个邪门的错没解决,所以先把文章大概内容告知: 1.spring框架aop注解扫描默认是关闭的,得手动开启. 2.关于Con't call commit when autocom ...
- Spring框架系列(11) - Spring AOP实现原理详解之Cglib代理实现
我们在前文中已经介绍了SpringAOP的切面实现和创建动态代理的过程,那么动态代理是如何工作的呢?本文主要介绍Cglib动态代理的案例和SpringAOP实现的原理.@pdai Spring框架系列 ...
- Spring框架系列(2) - Spring简单例子引入Spring要点
上文中我们简单介绍了Spring和Spring Framework的组件,那么这些Spring Framework组件是如何配合工作的呢?本文主要承接上文,向你展示Spring Framework组件 ...
- Spring框架系列(6) - Spring IOC实现原理详解之IOC体系结构设计
在对IoC有了初步的认知后,我们开始对IOC的实现原理进行深入理解.本文将帮助你站在设计者的角度去看IOC最顶层的结构设计.@pdai Spring框架系列(6) - Spring IOC实现原理详解 ...
- Spring框架系列(8) - Spring IOC实现原理详解之Bean实例化(生命周期,循环依赖等)
上文,我们看了IOC设计要点和设计结构:以及Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的:容器中存放的是Bean的定义即Be ...
- Spring框架系列(7) - Spring IOC实现原理详解之IOC初始化流程
上文,我们看了IOC设计要点和设计结构:紧接着这篇,我们可以看下源码的实现了:Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的. ...
随机推荐
- website 合集
1. oracle http://asktom.oracle.com ( 英文 ) http://itpub.net ( 中文 ) https://www.oracle.com/communitie ...
- android动画效果(转载)
一.动画基本类型: 如下表所示,Android的动画由四种类型组成,即可在xml中定义,也可在代码中定义,如下所示: XML CODE 渐变透明度动画效果 alpha AlphaAnimation 渐 ...
- Reducing the Dimensionality of Data with Neural Networks
****************内容加密中********************
- MFC绘图相关GDI工具对象和函数介绍
在利用MFC进行界面编程时,除了需要熟悉各种类型控件的操作外,还会经常遇到图形绘制和显示的问题,比如时频分析界面.图像处理界面等.处理这些软件界面开发问题时,不可避免地需要用到一系列GDI工具对象和相 ...
- deep learning+ Depth Estimation
Depth estimation/stereo matching/optical flow @CVPR 2017 Unsupervised Learning of Depth and Ego-Moti ...
- NLM算法
non-Local Means 非局部均值 论文原文:http://www.ipol.im/pub/art/2011/bcm_nlm/?utm_source=doi 论文源代码:http://www. ...
- Plug组件(不断跟新)
这个plug组件不知到底是什么东西,不知何com组件什么区别 #include <iostream> #include <plug/plug.h> #include " ...
- 云计算和SDN中的开源交换机介绍以及使用
之前关于SDN的开发工作都是在控制器层面上(以ryu为主),现在开始了新的工程项目,需要同时修改控制器和交换机的源码,如果后续项目需要,还可能需要加中间层——网络虚拟层,这部分的知识已经在前面读过了相 ...
- [LeetCode] Reverse Lists
Well, since the head pointer may also be modified, we create a new_head that points to it to facilit ...
- Golang Frameworks
Web frameworks help developers build applications as easily and quickly as possible. Go is still rel ...