Spring AOP深入剖析
一、通过代理工厂模式配置通知
①、前置通知、后置通知:
定义某接口:ISomeService,并自定义方法
public interface ISomeService {
public void tran() throws Exception;
public void log();
}
定义类 实现该接口,并重写方法:
public class SomeService implements ISomeService{
public void tran() throws Exception{
System.out.println("开启事务!!");
}
public void log() {
System.out.println("记录日志!!");
}
定义前置通知类,并实现MethodBeforeAdvice该接口
public class MyBefore implements MethodBeforeAdvice{
public void before(Method arg0, Object[] arg1, Object arg2)
throws Throwable {
System.out.println("==before==");
}
定义后置通知类,并实现AfterReturningAdvice该接口
public class MyAfter implements AfterReturningAdvice{
public void afterReturning(Object arg0, Method arg1, Object[] arg2,
Object arg3) throws Throwable {
System.out.println("==after==");
}
配置Spring配置文件applicationContext.xml:
代理工厂:ProxyFactoryBean

测试类:

实现效果:

② 环绕通知 MethodInterceptor
环绕增强在目标方法的前后都可以织入增强处理。环绕增强是功能最强大的强大处理。Spring把目标方法的控制权全部交给了他。在环绕增强处理中,可以获取或修改目标方法的参数、返回值、可以对它进行异常处理,甚至可以决定目标方法是否执行。

配置Spring文件:
<!-- 环绕增强 -->
<bean id="some" class="cn.happy.entity.SomeService"></bean> <bean id="arround" class="cn.happy.arround.MyInterceptor"></bean> <bean id="factory" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="some"></property>
<property name="interceptorNames" value="arround"></property>
</bean>
实现效果:

通过MethodInterceptor接口实现了环绕增强。该接口要求实现invoke()方法,其参数MethodInvocation不但封装目标方法及其参数组,还封装了被代理目标对象。通过proceed()方法可以调用目标对象的相应方法,从而实现对目标方法的完全控制!
③异常通知:
特点是在目标方法抛出异常时织入增强处理。通过ThrowsAdvice接口实现异常抛出增强,但ThrowsAdvice接口中并没有定义任何方法,但是我们在定义异常抛出的增强方法时必须遵守以下方法签名:
void afterThrowing([Method method,Object[]arguments,Object target,] Throwable ex)

实现类出现异常情况下:

Spring配置文件:
<!-- 异常通知 -->
<bean id="some" class="cn.happy.entity.SomeService"></bean> <bean id="throws" class="cn.happy.throwsAdvice.MyThrows"></bean> <bean id="factory" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="some"></property>
<property name="interceptorNames" value="throws"></property>
</bean>
测试类:
若将异常抛给上级处理,则在控制台通过,单测报错,若将异常手动抛出,则相反

@Test
public void proxyTest(){
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
ISomeService ser=(ISomeService) ctx.getBean("factory");
//ser.tran();
try {
ser.tran();
} catch (Exception e) {
e.printStackTrace();
}
ser.log(); }
二、顾问Advisor
顾问Advisor是Spring提供的另一种切面。其可以完成更为复杂的切面织入功能。PointcutAdvisor是顾问的一种,可以指定具体的切入点。顾问将通知进行了包装,会根据不同的通知类型,在不同的时间点,将切面织入到不同的切入点。
PointcutAdvisor接口有两个较为常用的实现类:
*:NameMatchMethodPointcutAdvisor 名称匹配方法切入点顾问
*: RegexpMethodPointcutAdvisor 正则表达式匹配方法切入点顾问
<property name="pattern" value=".*do.*"></property> 表示方法全名(包名,接口名,方法名)
运算符名称意义:
. 点号 表示任意单个字符
+ 加号 表示前一个字符出现一次或者多次
* 星号 表示前一个字符出现0次或者多次
如何实现:
同理:定义接口和实现类,并自定义方法。以及前置增强的类。关键点在Spring配置文件
①名称匹配方法切入点顾问

② 正则表达式匹配方法切入点顾问

三、自动代理生成器
注意:默认Advisor自动代理生成器,切面只能是顾问,对所有的对象都增强
两种实现方式:
① 默认Advisor自动代理生成器 DefaultAdvisorAutoProxyCreator
② BeanName自动代理生成器 BeanNameAutoProxyCreator
在这里 无需配置代理工厂bean,测试类getBean()取的id是配置文件的被代理对象
切面只能是顾问的情况下:

实现效果:

既可以是通知也可以是顾问的情况下:

实现效果:

测试类:
public class Test01 {
@Test
public void proxyTest(){
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
ISomeService ser=(ISomeService) ctx.getBean("some");
ser.tran();
ser.log();
}
四、Spring的经典AOP配置方案、使用的是Aspectj第三方框架,实现了AOP思想。注解配置的AOP,纯POJO <aop:config>
具体步骤:
① 在项目中添加SpringAOP相关的JAR文件
②使用注解定义前置增强和后置增强实现日志功能
③编写Spring配置文件,织入注解定义的增强
④编写代码获取带有增强处理的业务对象
核心JAR包:

实现思路:
1、定义接口实现类,并重写该方法
public interface ISomeService {
public void list();
}
public class SomeService implements ISomeService{
public void list() {
System.out.println("SomeService.list()");
}
}
2、通过注解实现增强,自定义类
使用@Aspect注解将该类定义为切面,并且使用@Before注解将该方法定义为前置增强,增强定义完后,就可以在Spring配置文件中织入使用注解定义的增强了
@Aspect
public class MyAspectj {
@Before(value = "execution(* *..service.*.*(..))")
public void MyBeforeAdvice(){
System.out.println("==before==");
}
}
3、Spring配置文件

4、进行测试:
public class Test01 {
@Test
public void proxyTest(){
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
ISomeService ser=(ISomeService) ctx.getBean("some");
ser.list();
}
实现效果:

※※※补充点:
切入点表达式:
execution(【modifiers-pattern?】 访问修饰符
ret-type-pattern 返回值类型
【declaring-type-pattern?】 全限定性类名
name-pattern(param-pattern) 方法名(参数名) 包名.类型名.方法名
【throws-pattern?】) 抛出异常类型
public void doLog(String log){
}
切入点表达式要匹配的对象就是目标方法的方法名。所以,execution表达式中明显就是方法的签名。
注意:表达式中加[]的部分表示可省略部分,各部分间用空格分开。在其中可以使用以下符号:
符号意义:
* 0至多个任意字符
.. 用在方法参数中,表示任意多个参数
用在包名后,表示当前包及其子包路径
+ 用在类名后,表示当前类及其子类
用在接口后,表示当前接口及其实现类
案例:
execution(public * *(..)) 指定切入点为:任意公共方法
execution(* set*(..)) 指定切入点为:任何一个以"set"开始的方法
Spring AOP深入剖析的更多相关文章
- Spring AOP 深入剖析
AOP是Spring提供的关键特性之一.AOP即面向切面编程,是OOP编程的有效补充.使用AOP技术,可以将一些系统性相关的编程工作,独立提取出来,独立实现,然后通过切面切入进系统.从而避免了在业务逻 ...
- [转载]Spring AOP 深入剖析
转载自 http://www.cnblogs.com/digdeep/p/4528353.html 多谢@digdeep AOP是Spring提供的关键特性之一.AOP即面向切面编程,是OOP编程的有 ...
- Spring AOP 实现原理与 CGLIB 应用
https://www.ibm.com/developerworks/cn/java/j-lo-springaopcglib/ AOP(Aspect Orient Programming),也就是面向 ...
- Spring AOP 实现原理与 CGLIB 应用--转
AOP(Aspect Orient Programming),作为面向对象编程的一种补充,广泛应用于处理一些具有横切性质的系统级服务,如事务管理.安全检查.缓存.对象池管理等.AOP 实现的关键就在于 ...
- spring AOP 之一:spring AOP功能介绍
一.AOP简介 AOP:是一种面向切面的编程范式,是一种编程思想,旨在通过分离横切关注点,提高模块化,可以跨越对象关注点.Aop的典型应用即spring的事务机制,日志记录.利用AOP可以对业务逻辑的 ...
- 【转】Spring AOP 实现原理与 CGLIB 应用
AOP(Aspect Orient Programming),作为面向对象编程的一种补充,广泛应用于处理一些具有横切性质的系统级服务,如事务管理.安全检查.缓存.对象池管理等.AOP 实现的关键就在于 ...
- spring AOP详解四
AOP(Aspect Orient Programming),作为面向对象编程的一种补充,广泛应用于处理一些具有横切性质的系统级服务,如事务管理.安全检查.缓存.对象池管理等.AOP 实现的关键就在于 ...
- Spring源码剖析6:Spring AOP概述
原文出处: 五月的仓颉 我们为什么要使用 AOP 前言 一年半前写了一篇文章Spring3:AOP,是当时学习如何使用Spring AOP的时候写的,比较基础.这篇文章最后的推荐以及回复认为我写的对大 ...
- Spring源码剖析7:AOP实现原理详解
前言 前面写了六篇文章详细地分析了Spring Bean加载流程,这部分完了之后就要进入一个比较困难的部分了,就是AOP的实现原理分析.为了探究AOP实现原理,首先定义几个类,一个Dao接口: pub ...
随机推荐
- C标准头文件<stdlib.h>
是个大杂烩,里面声明了从动态内存分配到常用算法等各种函数和宏 #数据类型 **size_t** **wchar_t** **div_t**是一个结构体类型,也是div()返回的类型 **ldiv_t* ...
- Runnable和Thread
1.Runnable是一个接口,当实现该接口时需要复用run方法,在run方法中实现自己的逻辑. 2.Thread是一个类,它其实实现了Runnable方法,也就是说当你通过new 一个Thread得 ...
- jquery实现表格动态添加
//点击追加触发$(function(){$("#button").click(function(){var div_ = $("#sel").val();va ...
- adb命令
一下是记录一些日常经常用的adb command, adb root: adb shell -> su -> return -> adb root(首先让安卓设备获得root权限,然 ...
- Atitit.异常处理 嵌套 冗长的解决方案
Atitit.异常处理 嵌套 冗长的解决方案 1. 异常处理的需要改进的地方1 2. +异常设计的初衷是, 在程序中出现错误时, 由程序自己处理错误, 尽量不要以exit(0)这种粗暴的方式中止程序 ...
- ASP.NET MVC+EF框架+EasyUI实现权限管理系列(23)-设置角色遗留问题和为权限设置角色以及EasyUI Tabs的使用
ASP.NET MVC+EF框架+EasyUI实现权限管系列 (开篇) (1):框架搭建 (2):数据库访问层的设计Demo (3):面向接口编程 (4 ):业务逻辑层的封装 ...
- mfc学习之路--如何删除通过控件新增的变量
刚刚学校mfc的人都会遇到这样一个问题(比如我),在照做书做一个mfc程序,给控件新增变量时变量类型错了,但是变量名对了,然后想要加个正确的时候提示"已经存在该对象",然后就傻了, ...
- js自建方法库(持续更新)
1.得到一个数,在一个有序数组中应该排在的位置序号: function orderInArr(num,arr) { if(num > arr[0]){ return 1 + arguments. ...
- sharepoint2013用场管理员进行文档库的爬网提示"没有权限,拒绝"的解决方法
爬网提示被拒绝,场管理员明明可以打开那个站点的,我初步怀疑是:环回请求(LoopbackRequest)导致的 解决方法就是修改环回问题.修改注册表 具体操作方法: http://www.c-shar ...
- 浅谈Bluetooth蓝牙开发
前言:项目用到蓝牙开发,具体蓝牙获取硬件传感器中的数据. 因为没有蓝牙开发的相关经验,决定先了解一些蓝牙开发的知识,再去看之前同事写的蓝牙相关代码. ------------------------- ...