spring in action学习札记 2 AOP的例子
spring in action学习笔记 2 AOP的例子
5.实现aop的例子
1.首先先来点预备类,咱定义一个表演的接口,代码如下:
就一个方法,表演节目,然后再定义2个实现类,ShowBoy和ShowGirl
这些要bean要让spring来帮我们管理,所以要把他们写到spring的配置文件中。现在先不写,一会统一写。
现在该干正事了,首先就是定义通知,也就是说,想在表演节目的时候插入什么事情呢?
我们定义一个观众类,让他们在表演的时候,做一些动作。
从这个类定义的方法大概可以看出,找座位和关手机应该是表演前发生的,鼓掌应该是表演后发生的,而要求退钱应该是表演发生意外后发生的。
总结一下,Spring的aop通知有5种形式
Before:org.springframework.aop.MethodBeforeAdvice,这个接口代表方法之前。
After-returning: org.springframework.aop.AfterReturningAdvice,这个代表返回后
After-throwing:org.springframework.aop.ThrowsAdvice,代表抛出异常后。
Around:org.aopalliance.intercept.MethodInterceptor,代表一个方法的周围。
Introduction:org.springframework.aop.IntroductionInterceptor,代表引入
现在来定义真正的通知,通知不是包含应该干什么和何时干吗,那就写把。
其中该干什么在 Audience中定义的,而什么时候,就是这些接口所实现的方法,带有before,after等。都表明了什么时候。
有了通知,就该定义切点了把,切点直接在配置文件里定义,这时,也顺便把通知和目标类一起定义到xml文件中。切点是干嘛的,切点是定义应该在哪些方法用切面的,他有2种定义方式,一种是用正则表达式,来匹配想要的方法,另一种是用aspectJ切点表达式。
<!--定义目标类,也就是想被织入通知的类-->
现在切点也有了,就要搞完整切面了,完整切面也叫通知者,直接在xml中配置.
切面有了,基本的事情都好了,现在可以想办法把切面用到目标类上了,aop的原理是通过代理实现的,所以我们要声明代理bean了。
首先,之前的ShowBoy和ShowGirl要改在xml中声明的id,他们现在是目标类,所以XML要改为
现在制作一个代理:
现在就把代理类伪装成showBoy了,为什么要这样,上一篇已经说过了。
现在测试一下
如果现在要showGirl也代理,那么就要写它的xml:
如果再有一个呢,那非给把你累死,发现定义代理,大部分代码是一样的,就是引用的目标类不同,这时就要把共有的代码抽出来,自己去写变换的那部分。
定义一个代理的父类,实现了2个参数,其中的目标类由自己写。
重新声明伪装类就简单很多了
但是,许多人还是觉得要定义这么一大堆东西,还是太麻烦,没错,我也觉得是麻烦,spring有更好的实现方式。明天再接着总结。
写的不错,不过应该吧xml文件的名称写清楚
所有的xml都写在spring-springcase.xml这个配置文件中
不会把,我用没问题啊~
我把你的配置文件复制下来就写了个单元测试...运行出错呢?不知道什么原因?
直接复制肯定是不行的,我那个配置文件是一点点改进的,要看一下把
我用的好像就是2.0的吧~
呵呵,多谢啊~
哈哈哈哈~~我也觉得
5.实现aop的例子
1.首先先来点预备类,咱定义一个表演的接口,代码如下:
public interface Perform { void perform(); }
就一个方法,表演节目,然后再定义2个实现类,ShowBoy和ShowGirl
public class ShowBoy implements Perform{ public void perform() { System.out.println("表演街舞"); } } public class ShowGirl implements Perform{ public void perform() { System.out.println("表演肚皮舞"); } }
这些要bean要让spring来帮我们管理,所以要把他们写到spring的配置文件中。现在先不写,一会统一写。
现在该干正事了,首先就是定义通知,也就是说,想在表演节目的时候插入什么事情呢?
我们定义一个观众类,让他们在表演的时候,做一些动作。
public class Audience { public Audience() { } public void takeSeat(){ System.out.println("观众们找到自己的座位,都坐下来了"); } public void turnOffMobilePhone(){ System.out.println("请所有观众确定手机已经关闭"); } public void appluad(){ System.out.println("观众们大声鼓掌,啪啦啪啦啪啦"); } public void demandRefund(){ System.out.println("演的太差了,我们要退钱!"); } }
从这个类定义的方法大概可以看出,找座位和关手机应该是表演前发生的,鼓掌应该是表演后发生的,而要求退钱应该是表演发生意外后发生的。
总结一下,Spring的aop通知有5种形式
Before:org.springframework.aop.MethodBeforeAdvice,这个接口代表方法之前。
After-returning: org.springframework.aop.AfterReturningAdvice,这个代表返回后
After-throwing:org.springframework.aop.ThrowsAdvice,代表抛出异常后。
Around:org.aopalliance.intercept.MethodInterceptor,代表一个方法的周围。
Introduction:org.springframework.aop.IntroductionInterceptor,代表引入
现在来定义真正的通知,通知不是包含应该干什么和何时干吗,那就写把。
public class AudienceAdvice implements MethodBeforeAdvice, AfterReturningAdvice, ThrowsAdvice{ private Audience audience; public void setAudience(Audience audience) { this.audience = audience; } public void before(Method method, Object[] objects, Object o) throws Throwable { audience.takeSeat(); audience.turnOffMobilePhone(); } public void afterReturning(Object o, Method method, Object[] objects, Object o1) throws Throwable { audience.appluad(); } public void afterThrowing(Throwable throwable){ audience.demandRefund(); } }
其中该干什么在 Audience中定义的,而什么时候,就是这些接口所实现的方法,带有before,after等。都表明了什么时候。
有了通知,就该定义切点了把,切点直接在配置文件里定义,这时,也顺便把通知和目标类一起定义到xml文件中。切点是干嘛的,切点是定义应该在哪些方法用切面的,他有2种定义方式,一种是用正则表达式,来匹配想要的方法,另一种是用aspectJ切点表达式。
<!--定义目标类,也就是想被织入通知的类-->
<bean id="showBoy" class="com.spring.springcase.ShowBoy"/> <bean id="showGirl" class="com.spring.springcase.ShowGirl"/> <!--定义了通知中的功能,此类做为通知的从属类--> <bean id="audience" class="com.spring.springcase.Audience"/> <!--定义通知--> <bean id="audienceAdvice" class="com.spring.springcase.AudienceAdvice"> <property name="audience" ref="audience"/> </bean> <!--定义切点,声明想要的方法:spring提供的定义切点方式--> <bean id="springpointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut"> <property name="pattern" value=".*perform"/> </bean> <!--定义切点,aspectJ定义的切点方式--> <bean id="asPectJpoincut" class="org.springframework.aop.aspectj.AspectJExpressionPointcut"> <property name="expression" value="execution(* Performer+.perform(..))"/> </bean>
现在切点也有了,就要搞完整切面了,完整切面也叫通知者,直接在xml中配置.
<!--定义完整切面,把定义好的切点和通知放进来就行了 spring定义方式--> <bean id="audienceAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"> <property name="pointcut" ref="springpointcut"/> <property name="advice" ref="audienceAdvice"/> </bean> <!--定义完整切面,aspectJ定义方式--> <bean id="audienceAdvisor" class="org.springframework.aop.aspectj.AspectJExpressionPointcutAdvisor"> <property name="expression" value="execution(* Performer+.perform(..))"/> <property name="advice" ref="audienceAdvice"/> </bean>
切面有了,基本的事情都好了,现在可以想办法把切面用到目标类上了,aop的原理是通过代理实现的,所以我们要声明代理bean了。
首先,之前的ShowBoy和ShowGirl要改在xml中声明的id,他们现在是目标类,所以XML要改为
<bean id="showBoyTarget" class="com.spring.springcase.ShowBoy"/> <bean id="showGirlTarget" class="com.spring.springcase.ShowGirl"/>
现在制作一个代理:
<!--定义代理类,第一个参数是要代理的对象,第2个是使用的切面,第3个是实现的接口--> <bean id="showBoy" class="org.springframework.aop.framework.ProxyFactoryBean"> <property name="target" ref="showBoyTarget"/> <property name="interceptorNames"> <list> <value>audienceAdvice</value> </list> </property> <property name="proxyInterfaces"> <list> <value>com.spring.springcase.Perform</value> </list> </property> </bean>
现在就把代理类伪装成showBoy了,为什么要这样,上一篇已经说过了。
现在测试一下
public class performTest extends TestCase{ ApplicationContext ctx; @Override protected void setUp() throws Exception { ctx = new ClassPathXmlApplicationContext("spring-springcase.xml"); } public void testShowBoy(){ Perform perform = (Perform)ctx.getBean("showBoy"); perform.perform(); } } 打印结果: 观众们找到自己的座位,都坐下来了 请所有观众确定手机已经关闭 表演街舞 观众们大声鼓掌,啪啦啪啦啪啦
如果现在要showGirl也代理,那么就要写它的xml:
<bean id="showGril" class="org.springframework.aop.framework.ProxyFactoryBean"> <property name="target" ref="showGirlTarget"/> <property name="interceptorNames"> <list> <value>audienceAdvice</value> </list> </property> <property name="proxyInterfaces"> <list> <value>com.spring.springcase.Perform</value> </list> </property> </bean>
如果再有一个呢,那非给把你累死,发现定义代理,大部分代码是一样的,就是引用的目标类不同,这时就要把共有的代码抽出来,自己去写变换的那部分。
定义一个代理的父类,实现了2个参数,其中的目标类由自己写。
<bean id="audienceProxyBase" class="org.springframework.aop.framework.ProxyFactoryBean" abstract="true"> <property name="interceptorNames"> <list> <value>audienceAdvice</value> </list> </property> <property name="proxyInterfaces"> <list> <value>com.spring.springcase.Perform</value> </list> </property> </bean>
重新声明伪装类就简单很多了
<bean id="showBoy" parent="audienceProxyBase"> <property name="target" ref="showBoyTarget"/> </bean> <bean id="showGirl" parent="audienceProxyBase"> <property name="target" ref="showGirlTarget"/> </bean>
但是,许多人还是觉得要定义这么一大堆东西,还是太麻烦,没错,我也觉得是麻烦,spring有更好的实现方式。明天再接着总结。
引用
写的不错,不过应该吧xml文件的名称写清楚
所有的xml都写在spring-springcase.xml这个配置文件中
26 楼
DoubleEO
2009-01-16
myprincejava 写道
DoubleEO 写道
myprincejava 写道
配置文件怎么老出错呢? it's mean ref=".*perform"
不会把,我用没问题啊~
我把你的配置文件复制下来就写了个单元测试...运行出错呢?不知道什么原因?
直接复制肯定是不行的,我那个配置文件是一点点改进的,要看一下把
27 楼
cgttian
2009-01-18
收藏收藏····
但是 我一直在找2.0的配置 为什么我的配置老是报异常呢 就加一个切面的事儿很简单才对,给果搞了半天还是不行。
能说说2.0的不??
但是 我一直在找2.0的配置 为什么我的配置老是报异常呢 就加一个切面的事儿很简单才对,给果搞了半天还是不行。
能说说2.0的不??
28 楼
DoubleEO
2009-01-18
cgttian 写道
收藏收藏····
但是 我一直在找2.0的配置 为什么我的配置老是报异常呢 就加一个切面的事儿很简单才对,给果搞了半天还是不行。
能说说2.0的不??
但是 我一直在找2.0的配置 为什么我的配置老是报异常呢 就加一个切面的事儿很简单才对,给果搞了半天还是不行。
能说说2.0的不??
我用的好像就是2.0的吧~
29 楼
lxz999
2009-01-19
写的不错,收藏了,呵呵
30 楼
liyun_1981
2009-01-20
哥们,说的言简意赅,朴素而透彻,我会继续关注你的此类文章。
By the way,看到你的头像我只有四个字:“功夫熊猫”,呵呵。
By the way,看到你的头像我只有四个字:“功夫熊猫”,呵呵。
31 楼
DoubleEO
2009-01-20
liyun_1981 写道
哥们,说的言简意赅,朴素而透彻,我会继续关注你的此类文章。
By the way,看到你的头像我只有四个字:“功夫熊猫”,呵呵。
By the way,看到你的头像我只有四个字:“功夫熊猫”,呵呵。
呵呵,多谢啊~
32 楼
towne
2009-02-01
execution表示执行方法时,* *表示任意返回类型,.perform表示perform方法,(..)表示任意参数设置
------------------
写文章写个半句 这习惯!
------------------
写文章写个半句 这习惯!
33 楼
haopeng_haopeng
2009-02-12
wonderful
すごいですね
すごいですね
34 楼
oldman
2009-02-16
很好的学习资料,关注中。。。
35 楼
xinlingdeqihang
2009-02-26
蛮好的,谢谢!
36 楼
k0363224
2009-02-26
# <!--定义切点,声明想要的方法:spring提供的定义切点方式-->
# <bean id="springpointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut">
# <property name="pattern" value=".*perform"/>
# </bean>
这句没有看懂, <property name="pattern" value=".*perform"/> “.*perform”是什么意思?
# <bean id="springpointcut" class="org.springframework.aop.support.JdkRegexpMethodPointcut">
# <property name="pattern" value=".*perform"/>
# </bean>
这句没有看懂, <property name="pattern" value=".*perform"/> “.*perform”是什么意思?
37 楼
xiaoyaozijacky
2009-02-26
iranger 写道
哥儿们你的照片老是让我想到太君。。。。。。。。。。
哈哈哈哈~~我也觉得
38 楼
kantery
2009-02-26
五种通知里面没有 Introduction:org.springframework.aop.IntroductionInterceptor,代表引入 吧,是:引入。
你少的写的是:after 最终通知 加上你的前面四种是这五种,呵呵。
你少的写的是:after 最终通知 加上你的前面四种是这五种,呵呵。
39 楼
lirenjiede
2009-02-26
不是有个MethodInterceptor吗,干嘛不用?
40 楼
lirenjiede
2009-02-26
<div class="quote_title">DoubleEO 写道</div>
<div class="quote_div">
<div class="quote_title">zookie 写道</div>
<div class="quote_div">楼主,换个头像吧</div>
<br />为什么?</div>
<p>容易让人拍砖</p>
<p> </p>
<div class="quote_div">
<div class="quote_title">zookie 写道</div>
<div class="quote_div">楼主,换个头像吧</div>
<br />为什么?</div>
<p>容易让人拍砖</p>
<p> </p>
41 楼
DoubleEO
2009-02-26
<div class="quote_title">lirenjiede 写道</div>
<div class="quote_div">
<div class="quote_title">DoubleEO 写道</div>
<div class="quote_div">
<div class="quote_title">zookie 写道</div>
<div class="quote_div">楼主,换个头像吧</div>
<br />为什么?</div>
<p>容易让人拍砖</p>
<p> </p>
</div>
<p> 那真的是德军的..</p>
<div class="quote_div">
<div class="quote_title">DoubleEO 写道</div>
<div class="quote_div">
<div class="quote_title">zookie 写道</div>
<div class="quote_div">楼主,换个头像吧</div>
<br />为什么?</div>
<p>容易让人拍砖</p>
<p> </p>
</div>
<p> 那真的是德军的..</p>
42 楼
caicop
2009-09-02
总结得很好!
打包和导入的包可以写一下啊,有时候用的类型在多个包里面都有,都不知道该用哪个呢!
打包和导入的包可以写一下啊,有时候用的类型在多个包里面都有,都不知道该用哪个呢!
43 楼
zhf07
2009-09-04
真的非常不错啊,谢谢分享
44 楼
昔日舞曲
2009-09-16
有个问题--
这里面的property为什么是:
而不是:
<!--定义完整切面,aspectJ定义方式--> <bean id="audienceAdvisor" class="org.springframework.aop.aspectj.AspectJExpressionPointcutAdvisor"> <property name="expression" value="execution(* Performer+.perform(..))"/> <property name="advice" ref="audienceAdvice"/> </bean>
这里面的property为什么是:
<property name="expression" value="execution(* Performer+.perform(..))"/>
而不是:
<property name="pointcut" ref="asPectJpoincut"/>
45 楼
lt0604
2009-12-15
设计太复杂了,希望能在简单点。