1.实现前置增强 必须实现接口MethodBeforeAdvice接口

创建对应的文件

public interface Animal {//主业务接口
void eat(); //目标方法
void sleep();
}

Animal接口

public class Dog implements Animal {//目标对象
/*
* 专心我们的逻辑
* 至于方法之前和方法之后要做的事情!我不关注!
* Ioc:降低了主业务之间的耦合度
* AOP:降低了主业务和交叉业务逻辑之间的耦合度
*/
@Override
public void eat() {
System.out.println("Dog在吃骨头");
} @Override
public void sleep() {
System.out.println("Dog睡觉");
} }

Dog实现类

public class AnimalBeforeAdvice implements MethodBeforeAdvice {
/**
* method:执行的方法
* args:方法的参数
* target:目标类对象
*/
@Override
public void before(Method method, Object[] args, Object target)
throws Throwable {
System.out.println("进入 了 前置 增强..............MethodBeforeAdvice");
} }

前置增强类

<?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:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"
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/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置 目标对象 -->
<bean id="dog" class="cn.bdqn.dao.impl.Dog"/> <!-- 配置前置 增强 -->
<bean id="before" class="cn.bdqn.advice.AnimalBeforeAdvice"/> <!-- 生成 代理对象 -->
<bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean">
<!-- 设置目标对象(被代理的对象)<property name="targetName" value="dog"/> -->
<property name="target" ref="dog"/>
<!-- 设置增强方式 -->
<property name="interceptorNames">
<value>before</value>
</property>
</bean> </beans>

applicationContext.xml文件

public class AnimalTest {

    @Test
public void test01(){
ApplicationContext context=
new ClassPathXmlApplicationContext("applicationContext.xml");
//使用代理对象进行操作
Animal animal=(Animal) context.getBean("proxyFactoryBean");
animal.eat();
System.out.println("************");
animal.sleep();
}
}

测试类

2.实现后置增强 必须实现接口AfterReturningAdvice接口

public class AnimalAfterAdvice implements AfterReturningAdvice {
/**
* returnValue:返回值 ,能获取返回值 但是如果对返回值进行了操作!没有意义!
* afterReturning()就没有返回值
*/
@Override
public void afterReturning(Object returnValue, Method method,
Object[] args, Object target) throws Throwable {
System.out.println("执行了 后置 通知.................AfterReturningAdvice");
} }

后置增强类

<?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:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"
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/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置 目标对象 -->
<bean id="dog" class="cn.bdqn.dao.impl.Dog"/> <!-- 配置前置 通知-->
<bean id="before" class="cn.bdqn.advice.AnimalBeforeAdvice"/>
<!-- 配置后置 通知-->
<bean id="after" class="cn.bdqn.advice.AnimalAfterAdvice"/> <!-- 生成 代理对象 -->
<bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean">
<!-- 设置目标对象(被代理的对象)<property name="targetName" value="dog"/> -->
<property name="target" ref="dog"/>
<!-- 设置增强方式 同时配置多个增强方式01 .value="before,after"-->
<property name="interceptorNames" >
<!-- 02. 使用array
<array>
<value>before</value>
<value>after</value>
</array> -->
<list>
<value>before</value>
<value>after</value>
</list>
</property>
</bean> </beans>

修改后的xml文件

测试代码不变 直接运行

3.实现环绕增强 必须实现接口MethodInterceptor接口

public class AnimalAroundAdvice implements MethodInterceptor {
/**
*
* 环绕增强 是在前置增强之后 和后置增强之前 执行!
* 可以对方法的返回值进行一系列的操作
*/
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("方法之前........MethodInterceptor");
Object result = invocation.proceed();
//判断是否有返回值
if (result!=null) {
result=((String)result).toUpperCase();
} System.out.println("方法之后........MethodInterceptor");
return result;
} }

环绕增强类

可以修改Animal中的任意一个方法带有返回值的进行测试! 如

public interface Animal {//主业务接口
String eat(); //目标方法
void sleep();
}

改变后的Animal接口

public class Dog implements Animal {//目标对象
@Override
public String eat() {
System.out.println("Dog在吃骨头");
return "abcd";
} @Override
public void sleep() {
System.out.println("Dog睡觉");
} }

改变后的Dog实现类

<?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:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"
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/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置 目标对象 -->
<bean id="dog" class="cn.bdqn.dao.impl.Dog"/>
<!-- 配置环绕 通知-->
<bean id="around" class="cn.bdqn.advice.AnimalAroundAdvice"/> <!-- 生成 代理对象 -->
<bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="dog"/>
<property name="interceptorNames" >
<list>
<value>around</value>
</list>
</property>
</bean> </beans>

xml文件

    //测试返回值
@Test
public void test02(){
ApplicationContext context=
new ClassPathXmlApplicationContext("applicationContext.xml");
//使用代理对象进行操作
Animal animal=(Animal) context.getBean("proxyFactoryBean");
System.out.println(animal.eat());
System.out.println("************");
animal.sleep();
}

测试类

效果图

4.实现自定义增强

并不是所有的方法都需要增强,那么我们就需要自定义增强的方法

<?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:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"
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/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置 目标对象 -->
<bean id="dog" class="cn.bdqn.dao.impl.Dog"/> <!-- 配置前置 通知-->
<bean id="before" class="cn.bdqn.advice.AnimalBeforeAdvice"/> <!-- 自定义通知的切入点 -->
<bean id="myAdvice" class="org.springframework.aop.support.NameMatchMethodPointcutAdvisor">
<property name="advice" ref="before"></property>
<property name="mappedNames">
<list>
<value>eat</value> <!-- eat之外的方法就不会有前置通知了 -->
</list>
</property>
</bean> <!-- 生成 代理对象 -->
<bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="target" ref="dog"/>
<property name="interceptorNames" >
<list>
<value>myAdvice</value> <!--引入自定义增强 -->
</list>
</property>
</bean> </beans>

修改后的xml文件

测试代码不变

spring06Aop的更多相关文章

随机推荐

  1. 『重构--改善既有代码的设计』读书笔记----Inline Method

    加入间接层确实是可以带来便利,但过多的间接层有时候会让我自己都觉得有点恐怖,有些时候,语句本身已经够清晰的同时就没必要再嵌一个函数来调用了,这样只会适得其反.比如 void test() { if ( ...

  2. 安卓webview下使用zepto的swipe失效

    安卓webview下使用zepto的swipe遇到的坑 众所周知,安卓手机上touch事件一直有各种各样莫名其妙的问题. 比如,我想要用swipeLeft/swipeRight监听向左向右滑动事件,如 ...

  3. python抓取网页图片

    本人比较喜欢海贼王漫画,所以特意选择了网站http://www.mmonly.cc/ktmh/hzw/list_34_2.html来抓取海贼王的图片. 因为是刚刚学习python,代码写的不好,不要喷 ...

  4. C++学习笔记3——类的封装(1)

    封装: 1.把属性和方法进行封装. 2.对属性和方法进行访问控制. class和struct的区别: class和struct的唯一区别是默认的访问权限不一样.struct的默认访问权限是public ...

  5. SQL Server分区动态生成脚本(三)(按年份划分)

    --生成分区脚本DECLARE @DataBaseName NVARCHAR(50)--数据库名称DECLARE @TableName NVARCHAR(50)--表名称DECLARE @Column ...

  6. 至芯FPGA培训中心-1天FPGA设计集训(赠送FPGA开发板)

    至芯FPGA培训中心-1天FPGA设计集训(赠送开发板) 开课时间2014年5月3日 课程介绍 FPGA设计初级培训班是针对于FPGA设计技术初学者的课程.课程不仅是对FPGA结构资源和设计流程的描述 ...

  7. Sample RWD Setup for Client-Side Development

    RWD and RESS concepts fluid images, responsive grids, and media queries. Twitter's Bootstrap based o ...

  8. dnat,snat

    Iptables实现NAT是最基本的功能,大部分家用路由都是基于其SNAT方式上网,使用Iptables实现外网DNAT也很简单,不过经常会出现不能正常NAT的现象. 以下命令将客户端访问1.,很多人 ...

  9. 求帮看!!!!BZOJ 1014 [JSOI2008]火星人prefix

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4164  Solved: 1277[Submit] ...

  10. -_-#【Better Code】字符串匹配

    提高 web 应用性能之 JavaScript 性能调优