spring中AOP的实现就是通过动态代理来实现的。动态代理的实现在上篇blog中已经涉及。Spring中目前最为实用的AOP应用,非用其实现的事务管理机制莫属。也正是这一点,使得Spring AOP大方异彩。
# x1 x8 X5 r' k那么我们继续围绕上节的例子来探讨一下Spring中AOP机制的应用与开发。 首先,看看AOP中几个基本的概念对应在AOP中的实现:
( [; q# I* j( }. Q( n? 切点(PointCut) ! B: h0 s2 v7 @8 x U
一系列连接点的集合,它指明处理方式(Advice)将在何时被触发。
$ l$ z+ `. z' q! ] p) t7 W 对于我们开发而言,“何时触发”的条件大多是面向Bean的方法进行制定。像Spring的配置化事务管理时针对方法名称可进行PointCut设置,从而指定对所有以声明字符开头的方法进行基于AOP的事务管理。那么同样,对于我们自己实现的AOP组件而言,我们也可以以方法名作为触发判定条件。
4 C% ~1 Y, `, A1 @3 q7 {( l; g我们可以通过在XML配置文件中配置以下节点,为我们的组件设定触发条件。 g" Y, _+ W" q" V: f' x# G9 h6 q$ o
<bean id="myPointcutAdvisor" 1 t5 A0 B+ E+ C* D/ P6 M% z2 z
class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"> 0 H/ f) u( U" @( U
<property name="advice">
2 S) I0 Y' ~/ i- q+ k, D' Y<ref local="MyInterceptor" /> 5 T5 I# m0 r/ o# f0 F$ M4 q6 x
</property>
+ M" v3 `6 T4 ^% U; D1 |<property name="patterns">
* d: u2 c0 C6 |& Z<list>
3 U: d0 R" h# @2 }2 j<value>.*save.*</value> 5 b9 W2 H4 E" R
<value>.*do.*</value>
7 v$ Q8 u5 Z4 d4 G: P</list>
& Q& Z& {$ e+ m! {4 M- U1 n. V* L</property> " C* L7 _5 e$ H
</bean> 上面我们针对MyInterceptor设定了一个基于方法名的触发条件,也就是说,当目标类的指定方法运行时,MyInterceptor即被触发。 % @) V0 s ?# j
? 处理方式(Advice)
8 x; Q. J% B+ @3 O 也就是说要实现一个Interceptor,以供在连接点时触发。Spring中采用了AOP联盟(AOP Alliance)的通用AOP接口(接口定义位aopalliance.jar)。这里我们采用aopalliance.jar中定义的MethodInterceptor作为我们的Advice实现接口。那么我们上节例子中的处理方式应实现为如下Interceptor:
* n5 F* W+ ?5 t/ G. D4 F3 hpackage test.aop.spring; + K; ]+ s" z9 @( \1 S, I
import java.util.logging.Logger; : z( u# Z5 _' e; E5 f* M6 n' v8 Q
import org.aopalliance.intercept.MethodInterceptor; L1 k, @9 r5 `$ K" i; k1 Q
import org.aopalliance.intercept.MethodInvocation; /** , O0 C5 S& V* C, Q$ o1 \, H
* @author xkf
) X" Z) H: C" _**/ * `" ^8 n2 I# i# _1 u! _7 R) p6 f- P
public class LockInterceptor implements MethodInterceptor { & u8 R5 f: T) [! S, r
private Logger logger = Logger.getLogger(this.getClass().getName());
7 ]+ f2 F& o9 p( J6 gpublic Object invoke(MethodInvocation invocation) throws Throwable {
4 ~- K" ?; q" r# y0 U: H; P// TODO Auto-generated method stub
- [: }. q/ ~: {: |0 v3 hlock(); 2 ~5 Z/ o, W, [# g$ Y; y
Object ret= invocation.proceed();
( B$ B; H# s' t4 X% z) [0 gunlock();
9 D4 C, v+ }& Y3 t. ereturn ret; ) l2 G- A. {8 C+ j# L5 {
}
0 E% u* ]3 {4 X- D' Jprivate void lock(){
& w: _$ J6 ?$ D0 P: M" i3 P2 Mlogger.info("lock domain object..."); 5 H/ l; T7 ~$ l& a9 ?1 i, w
}
: f0 S u$ x8 n/ S& k, M5 Yprivate void unlock(){
+ [/ Y' E. e) m3 Klogger.info("unlock domain object...");
, O( `& }% `# r4 G _% M* h} # `0 [7 k: M* }2 u* y' w
}
. E+ ?0 \( Q3 d5 S$ x 实现后,对应的Interceptor实现类在配置文件中的体现如下: 0 f4 I3 F8 |* |+ Y' t
<bean id="MyInterceptor" class="test.aop.spring.LockInterceptor"/> * ~; G( ?) U2 H, Z5 g, D/ y1 ?
最后,我们还需要定义一个Spring AOP ProxyFactory用于加载执行AOP组件,并且需要将Advice通过IOC的方式注入到接口以及实现类。 0 R% w5 A8 P: M$ ~$ U: V
对应的配置文件应如下配置从而实现这些内容:
3 M" X. N6 d3 W<bean id="myAOPProxy" class="org.springframework.aop.framework.ProxyFactoryBean">5 Q0 A9 o0 B6 Q5 i% R ^, G M9 F2 s1 i! R
<property name="proxyInterfaces">
+ h) H) F( S' P% ]- Z% s<value>test.aop.spring.DomainObjDAO</value>
. l' N! a+ o, m2 Y G: N7 U+ {: Z6 l</property> # _1 C Z+ E" u( G0 O
<property name="target">
$ d# B+ H5 u6 }7 \( t<ref local="impl" /> , m: n/ @$ A1 S( X
</property>
7 h% b4 a5 D. o6 K2 M<property name="interceptorNames">
0 y$ S0 O% l; \6 P<value>myPointcutAdvisor</value> # u+ C; a( o+ R# q
</property>
- U; V( ~ K% J% \2 v</bean>
8 z& F+ d6 x$ F! B9 {0 v<bean id="impl" class="test.aop.spring.DomainObjDAOImpl"/> * e6 A$ W6 m* l
万事大吉!写一个TestCase来看一下运行结果: + u3 B3 |, n$ b$ v
package test.aop.spring; import junit.framework.TestCase;
! R% ?* T; t2 {& J2 M0 Pimport org.springframework.context.ApplicationContext;
6 D1 C4 v6 Y5 X, J1 m% ^( @import org.springframework.context.support.FileSystemXmlApplicationContext; /**
$ C/ c* G: O1 n& E* @2 t* @author xkf
# y2 \1 @5 Y6 g/ W0 ~# o3 `( E) v, O2 e*/
" A9 Q% R% F [2 Q" [( fpublic class SpringTestCase extends TestCase {
* s2 T/ y$ i3 {# o1 y$ F# r3 `DomainObjDAO test=null; protected void setUp() throws Exception { ( T# i! a( d( T5 j1 X
super.setUp(); 5 S% G2 \) {2 o; }: ]
ApplicationContext ctx=new FileSystemXmlApplicationContext("test/Bean.xml");
$ L9 |0 Y$ Z- r7 ttest = (DomainObjDAO) ctx.getBean("myAOPProxy"); ( I7 D/ U2 i( f
} protected void tearDown() throws Exception {
1 v6 \! m, t$ \& R0 v' psuper.tearDown(); ! ^( e. W; Y7 u0 C9 X! H% \
}
# P9 O# Y. v0 X8 Gpublic void testSave(){ 0 K; u2 N4 _6 i0 S7 |
test.save();
* [9 i; F# X; d6 J} 7 a8 [+ @7 e+ Y, J0 E2 A* Z
}
* p o' ?! {( h运行结果如下:
8 J% }; r" W6 b/ e8 A信息: Creating shared instance of singleton bean ''MyInterceptor''
! q) g& V8 q. n) [ ]3 ~2004-12-7 23:50:11 test.aop.spring.LockInterceptor lock & ^/ W, k; k' x( V$ {/ B
信息: lock domain object... 8 ]& `+ [9 Z- U8 X' P0 C0 R5 C
saving domain object......
7 y* q& e3 U7 Q! Z: M7 B( D2004-12-7 23:50:11 test.aop.spring.LockInterceptor unlock
% B: E" p6 P- b& M信息: unlock domain object... OK!至此,我们已经探讨了AOP动态代理的实现以及其在Spring中的应用,接下来的Blog中我将会关注Spring中事务以及持久化的实现。 $ \% g1 P. F6 j
|