什么是AOPQ?/p>
一个方面(aspect)是一个共有的Ҏ,h此代表性的有:横向分离的方法,c,对象层次或者实体对象模型。它们看h应该是组合在一LQ但是在AOP里面你不用像以前面对对象QOOQ那hl织它们了?/p>
在传lJAVA中要加入计算旉的代码到你的应用中,你必L以下方式Q?/p>
public class BankAccountDAO{
public void withdraw(double amount){
long startTime = System.currentTimeMillis();
try {
// Actual method body...
}
finally {
long endTime = System.currentTimeMillis() - startTime;
System.out.println("withdraw took: " endTime);
}
}
}
我们可以列Dq里面存在的几个问题Q?/p>
1。如果你要在每个你的Ҏ中都加入q样的代理,无疑是十分糟p的Q特别是有try/catchq样的语句?/p>
2。这里有很多代码都不是你真实需要用到的Q这样就使你的程序代码十分臃肿,读v来也十分困难。而且你不得不把你的代码放在try里面......
3。如果你要扩展这D代码,我们可以遇见到那是十分困难的工作?/p>
所以可见,q样的代码是十分难于l护Q扩展和l承的,因ؓ在这里面有很多东西分散了你对你这D代码真正要实现的东西的注意力。而且q只不过是一D|单的例子Q在真正的OOP中是很难实现对以上代码更好的表现Ҏ的?/p>
面对斚w~程可以分离你的q些功能性,可以让你增加行ؓ来围l你的功能代码。例如上面的QAOP可以在执行你自己的代码前你可以控制执行其它的功能?/p>
所有实现AOP的框枉有两U方式:l装x??a programmatic construct(~程实现)
JBOSS的一个横向切入关注点
01. public class Metrics implements org.jboss.aop.Interceptor
02. {
03. public Object invoke(Invocation invocation) throws Throwable
04. {
05. long startTime = System.currentTimeMillis();
06. try
07. {
08. return invocation.invokeNext();
09. }
10. finally
11. {
12. long endTime = System.currentTimeMillis() - startTime;
13. java.lang.reflect.Method m = ((MethodInvocation)invocation).method;
14. System.out.println("method " m.toString() " time: " endTime "ms");
15. }
16. }
17. }
真正实现的功能代码是?行调用了Q这是实现了组装关注点Q之成Z一个方面。这让我们在以后扩展实际功能的时候就十分方便了,只需要去修改具体的实现方法,而不用去兛_其它x点了?/p>
JBOOS中具体应用这个方?/p>
需要定义一个切入点(pointcuts)Q全部通过政则表达式来实现?/p>
Listing Three: Defining a pointcut in JBoss AOP
1. <bind pointcut="public void com.mc.BankAccountDAO->withdraw(double amount)">
2. <interceptor class="com.mc.Metrics"/>
3. </bind >
4. <bind pointcut="* com.mc.billing.*->*(..)">
5. <interceptor class="com.mc.Metrics"/>
6. </bind >
1-3定义的一个切入点的方法就?BankAccountDAO->withdraw(double amount)
4-6定义的是一个通用的,它的切入Ҏ所有的com.mc.billing.下面的类的方法?/p>
既然我们知道了需要用advisor向主要代码中注入“不可见的”adviceQ让我们实现一个Spring AOP的例子。在q个例子中,我们实C个before adviceQ这意味着advice的代码在被调用的publicҎ开始前被执行。以下是q个before advice的实C码:
代码:
package com.company.springaop.test;
import java.lang.reflect.Method;
import org.springframework.aop.MethodBeforeAdvice;
public class TestBeforeAdvice implements MethodBeforeAdvice {
public void before(Method m, Object[] args, Object target)
throws Throwable {
System.out.println("Hello world! (by "
+ this.getClass().getName()
+ ")");
}
}
接口MethodBeforeAdvice只有一个方法before需要实玎ͼ它定义了advice的实现。beforeҎq三个参数Q它们提供了相当丰富的信息。参数Method m是advice开始后执行的方法。方法名U可以用作判断是否执行代码的条g。Object[] args是传l被调用的publicҎ的参数数l。当需要记日志Ӟ参数args和被执行Ҏ的名Uͼ都是非常有用的信息。你也可以改变传lm的参敎ͼ但要心使用q个功能Q编写最初主E序的程序员q不知道ȝ序可能会和传入参数的发生冲突。Object target是执行方法m对象的引用?/p>
在下面的BeanImplcMQ每个publicҎ调用前,都会执行adviceQ?/p>
代码:
package com.company.springaop.test;
public class BeanImpl implements Bean {
public void theMethod() {
System.out.println(this.getClass().getName()
+ "." + new Exception().getStackTrace()[0].getMethodName()
+ "()"
+ " says HELLO!");
}
}
cBeanImpl实现了下面的接口BeanQ?/p>
代码:
package com.company.springaop.test;
public interface Bean {
public void theMethod();
}
虽然不是必须使用接口Q但面向接口而不是面向实现编E是良好的编E实践,Spring也鼓p样做?/p>
pointcut和advice通过配置文g来实玎ͼ因此Q接下来你只需~写L法的Java代码Q?/p>
代码:
package com.company.springaop.test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.FileSystemXmlApplicationContext;
public class Main {
public static void main(String[] args) {
//Read the configuration file
ApplicationContext ctx = new FileSystemXmlApplicationContext("springconfig.xml");
//Instantiate an object
Bean x = (Bean) ctx.getBean("bean");
//Execute the public method of the bean (the test)
x.theMethod();
}
}
我们从读入和处理配置文g开始,接下来马上要创徏它。这个配|文件将作ؓ_合E序不同部分的“胶水”。读入和处理配置文g后,我们会得C个创建工厂ctx。Q何一个Spring理的对象都必须通过q个工厂来创建。对象通过工厂创徏后便可正怋用?/p>
仅仅用配|文件便可把E序的每一部分l装h?/p>
代码:
AspectWerkz 主要Ҏ:
1Q运行时和加载时字节码修正:你可以在q行时或~译时轻杄攚wQ何(旧)应用E序或除?span lang="EN-US">rt.jar以外的外部类?span lang="EN-US">
2Q支?span lang="EN-US">join point模型
3Q支?span lang="EN-US">AnnotationQ匹?span lang="EN-US">JavaDoc?span lang="EN-US">JSR-175Q支持用戯定义Annotation
4Q支持部|多?span lang="EN-US">Aspect定义文g到部|的应用E序Q?span lang="EN-US">WEB-INF/aop.xml?span lang="EN-US">META-INF/aop.xmlQ?span lang="EN-US">
5Q?span lang="EN-US">Introduction/内类型声明(也称MixinQ,也就是具有添加接口和实现到已存在的类中的能力
6Q?span lang="EN-US">Annotation定义Q定?span lang="EN-US">Aspect使用的运行时AnnotationQؓJSR-175准备Q?span lang="EN-US">
7Q?span lang="EN-US">XML定义Q定?span lang="EN-US">Aspect使用?span lang="EN-US">XMLQ?span lang="EN-US">XML可以用来_、改写和解析Annotation定义
8Q插件式Aspect理器能够和IoC框架Q如Spring?span lang="EN-US">PicoContainerQ一起工?span lang="EN-US">
9Q四U不同的Advice?span lang="EN-US">Introduction部v模型Q范_Q?span lang="EN-US">perJVMQ单模式Q?span lang="EN-US"> perClass?span lang="EN-US">perInstance ?span lang="EN-US">perThread
10Q?span lang="EN-US">Advice?span lang="EN-US">Introduction能够动态部|Ӏ反部v或重新部|?span lang="EN-US">
11Q高性能Q?span lang="EN-US">JIT~译
12Q?span lang="EN-US">Fine-grained模式语言选择join point
13Q所?span lang="EN-US">Advice能够和所有的join point和各U合类型的pointcut
14Q脱机变换(可以用作后处理器Q?span lang="EN-US">
15Q?span lang="EN-US">Aspect?span lang="EN-US">Advice?span lang="EN-US">Introduction使用POJO~码
16Q目标类可以是正规的POJOQ也是不需要接?span lang="EN-US">
17Q支持通过定义传递参数给Advice和定义可重用?span lang="EN-US">Advice堆栈
18Q元数据被加到类?span lang="EN-US">
19Q简单的用法和配|?span lang="EN-US">
开?span lang="EN-US">AOP
1Q这里我们要在屏q打印出?span lang="EN-US">Hello AOP!”,看如下代码:
//HelloAOP.java
public class HelloAOP {
public static void main(String args[]) {
HelloAOP ha = new HelloAOP();
ha.test();
}
public void test() {
System.out.println("Hello AOP!");
}
}
~译HelloAOP.java文gQ?span lang="EN-US">javac HelloAOP.java
2Q现在我要在输出?span lang="EN-US">Hello AOP!”前后做一些工作,q些工作在运行时会得到调用机会,如果使用AOP术语Q我们可以说我们要编写我们的aspectQ这?span lang="EN-US">aspect会在q行时被weave into Q织入)HelloAOP class?span lang="EN-US">
//MyAspect.java
import org.codehaus.aspectwerkz.joinpoint.JoinPoint;
public class MyAspect {
public void beforeTesting(JoinPoint joinPoint) {
System.out.println("before testing...");
}
public void afterTesting(JoinPoint joinPoint) {
System.out.println("after testing...");
}
}
javac MyAspect.java
3Q织入过Eƈ不简单,我们需要撰写一个描q文件来?span lang="EN-US">aspect和其l入?span lang="EN-US">class中的信息联系h?span lang="EN-US">
//aop.xml
<aspectwerkz>
<system id="AspectWerkzExample">
<aspect class="MyAspect">
<pointcut name="testMethod" expression="execution(* HelloAOP.test(..))"/>
<advice name="beforeTesting" type="before" bind-to="testMethod"/>
<advice name="afterTesting" type="after" bind-to="testMethod"/>
</aspect>
</system>
</aspectwerkz>
4)run it
aspectwerkz -Daspectwerkz.definition.file=aop.xml HelloAOP
//output:
before testing...
Hello AOP!
after testing...