spring中AOP的实现就是通过动态代理来实现的。动态代理的实现在上篇blog中已经涉及。Spring中目前最为实用的AOP应用,非用其实现的事务管理机制莫属。也正是这一点,使得Spring AOP大方异彩。
" E$ T7 M2 v- h4 K那么我们继续围绕上节的例子来探讨一下Spring中AOP机制的应用与开发。 首先,看看AOP中几个基本的概念对应在AOP中的实现:
: s( L) W0 Y" [7 c" a, W$ c? 切点(PointCut)
% o, n+ X; M- z; p" U) h$ y 一系列连接点的集合,它指明处理方式(Advice)将在何时被触发。 , Z. L( m) s- F' J
对于我们开发而言,“何时触发”的条件大多是面向Bean的方法进行制定。像Spring的配置化事务管理时针对方法名称可进行PointCut设置,从而指定对所有以声明字符开头的方法进行基于AOP的事务管理。那么同样,对于我们自己实现的AOP组件而言,我们也可以以方法名作为触发判定条件。2 B4 K8 {' s3 x7 A, M; S- r0 g
我们可以通过在XML配置文件中配置以下节点,为我们的组件设定触发条件。 W. P) |) w/ v$ b) w0 [! d
<bean id="myPointcutAdvisor" / f; _ y2 K( w+ J" j! ?
class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> : A* h/ H5 K& G* ~3 ]
<property name="advice">
4 H2 b% e! K+ ?+ D0 s2 r/ m7 S<ref local="MyInterceptor" /> ( T3 _; N7 N: S z# {# k' r
</property> 0 P r# b, j( H3 A/ m8 D3 e& p
<property name="patterns">
! r5 w8 s& g8 O9 v$ B<list>
2 u( D: h% c, g: w1 A* m<value>.*save.*</value> 1 Z, k$ ~/ G9 k. b0 z$ g
<value>.*do.*</value> 3 ]' O/ N, d u* A9 E! u6 d3 [! F
</list>
0 T9 G, a' _ g/ f5 m, Q5 A</property> 0 `+ E9 B3 w! _
</bean> 上面我们针对MyInterceptor设定了一个基于方法名的触发条件,也就是说,当目标类的指定方法运行时,MyInterceptor即被触发。
, m! Q1 p# j z* G4 H4 h. n ? 处理方式(Advice)
: j& m$ C% @9 D$ [, m- P 也就是说要实现一个Interceptor,以供在连接点时触发。Spring中采用了AOP联盟(AOP Alliance)的通用AOP接口(接口定义位aopalliance.jar)。这里我们采用aopalliance.jar中定义的MethodInterceptor作为我们的Advice实现接口。那么我们上节例子中的处理方式应实现为如下Interceptor:- @& }( n0 J/ l" C5 r* h3 C h! ~
package test.aop.spring;
6 S; Y, F& S3 k" Limport java.util.logging.Logger;
: J' j* n' o6 a* K- O: i- Himport org.aopalliance.intercept.MethodInterceptor; ) \* @7 k& n! L$ F) R9 z9 ~. g6 B; f
import org.aopalliance.intercept.MethodInvocation; /** Z" v" _0 s1 ~/ ]
* @author xkf ' [. h3 \( m; |3 y& Y+ p9 ]
**/
v5 C% F. F; f6 c) g5 fpublic class LockInterceptor implements MethodInterceptor {
2 z+ w; F% s; G" b9 T( Y) Xprivate Logger logger = Logger.getLogger(this.getClass().getName());
' }: d- i; D( P/ R& I/ N% Gpublic Object invoke(MethodInvocation invocation) throws Throwable {
|$ _2 d, x n8 H1 R5 p# J// TODO Auto-generated method stub
; D0 ? F% Z% n5 [( Hlock(); & C/ ~8 ^ {9 t# v. F. C
Object ret= invocation.proceed(); 4 V/ y* d, M2 L; c- d
unlock(); $ C/ A3 S h2 E& e8 h
return ret;
3 V/ a/ o* {1 v( e} : s$ B6 i; j* U; M4 e: U
private void lock(){
( v+ D$ e1 e1 E0 i4 Y8 Q. A5 Dlogger.info("lock domain object..."); . W. U, K9 R* _( W& I
}
& I0 z! K4 O- X% b4 _private void unlock(){ 6 l+ L& `' t2 X; h; e
logger.info("unlock domain object...");
" a* o$ s0 b9 O/ y0 Y/ R}
5 a) T3 R7 }2 J5 e3 I- J3 K$ G} % o b$ R7 z+ x" u# K+ b4 v
实现后,对应的Interceptor实现类在配置文件中的体现如下: + [9 Z! m! a% }. ?: D
<bean id="MyInterceptor" class="test.aop.spring.LockInterceptor"/>
( J& X" \3 }. g9 j' l6 f9 x 最后,我们还需要定义一个Spring AOP ProxyFactory用于加载执行AOP组件,并且需要将Advice通过IOC的方式注入到接口以及实现类。
) q8 c6 | d6 M" B; E 对应的配置文件应如下配置从而实现这些内容: ! X- F7 j8 c3 p5 `. w" M* n; f5 H* A
<bean id="myAOPProxy" class="org.springframework.aop.framework.ProxyFactoryBean"># N2 g( M# z; {/ t
<property name="proxyInterfaces">
2 J9 j+ u* { b5 j* i<value>test.aop.spring.DomainObjDAO</value>
4 R3 A$ h- \2 y</property>
- x4 k O2 S. C I' \<property name="target">
G, r# w1 ]" H8 {: v<ref local="impl" />
8 n* X. r% g3 \1 |( w% S/ l9 _5 ^/ ^ L2 Q</property>
1 m: L, Q- F5 a0 b' ]; _) M<property name="interceptorNames">
% ]* _8 P5 M( l! m& w<value>myPointcutAdvisor</value> 5 j5 E7 J, m2 h' ^; s; S( S2 A4 r. E$ C
</property>
7 F) |8 }# E, x3 A</bean> 2 \8 M, s! N; y6 M3 m$ G
<bean id="impl" class="test.aop.spring.DomainObjDAOImpl"/>
0 J" j; H; B ~5 w% I! k- O, C万事大吉!写一个TestCase来看一下运行结果:
9 I; J4 G( p1 T* P9 ?* I$ G. x6 Cpackage test.aop.spring; import junit.framework.TestCase; 4 n: i' ?# I9 Q8 {0 S* B
import org.springframework.context.ApplicationContext;
5 g- ^4 W- K: h7 t" I+ _5 J* eimport org.springframework.context.support.FileSystemXmlApplicationContext; /** 6 i) }* c; \1 t, a# E' P/ R. r& m
* @author xkf 1 ~- ?4 P/ O, z) `( D$ p
*/ 8 q2 n. Y0 `: V% y ^
public class SpringTestCase extends TestCase { 7 K" v$ s5 [# q: D4 \, f* m( H
DomainObjDAO test=null; protected void setUp() throws Exception {
) t5 S L; I2 J1 `3 Esuper.setUp();
0 A- P4 S. _4 m' C- dApplicationContext ctx=new FileSystemXmlApplicationContext("test/Bean.xml");
' K! ~6 ~+ Z2 ]! N$ ytest = (DomainObjDAO) ctx.getBean("myAOPProxy"); + h3 n8 J4 r# |6 e5 ?
} protected void tearDown() throws Exception {
. t$ f& A' H. ^super.tearDown(); 1 X( C- k0 P/ @& u' b m- K
}
& X/ [! o5 T% J4 d" |8 c/ ?public void testSave(){
% l- C, L* x# x+ t; Ntest.save();
7 D* ~1 F0 s4 }2 `} 3 k9 G8 m; ^0 C0 S& `
} : _9 [3 u t: p4 l( B9 {6 q
运行结果如下: % L k9 N3 h" Y$ e4 G n+ x: u# u7 N
信息: Creating shared instance of singleton bean ''MyInterceptor''
' p K' f; s* ?, a6 u3 F8 o1 E2004-12-7 23:50:11 test.aop.spring.LockInterceptor lock ! I5 k7 u* w3 N7 w7 F0 N" p6 n. j
信息: lock domain object... , p: T: w5 M1 P: |1 c |
saving domain object......
# l7 s0 t5 E, F( E2004-12-7 23:50:11 test.aop.spring.LockInterceptor unlock
. R$ A( C) F( X. a. L信息: unlock domain object... OK!至此,我们已经探讨了AOP动态代理的实现以及其在Spring中的应用,接下来的Blog中我将会关注Spring中事务以及持久化的实现。 5 [0 |1 G8 n& |$ o1 a
|