AspectJ简介

AOP虽然是方法论,但就好像OOP中的Java一样,一些先行者也开发了一套语言来支持AOP。目前用得比较火的就是AspectJ了,它是一种几乎和Java完全一样的语言,而且完全兼容Java(AspectJ应该就是一种扩展Java,但它不是像Groovy那样的拓展)。当然,除了使用AspectJ特殊的语言外,AspectJ还支持原生的Java,只要加上对应的AspectJ注解就好。所以,使用AspectJ有两种方法:

1.完全使用AspectJ的语言。这语言一点也不难,和Java几乎一样,也能在AspectJ中调用Java的任何类库。AspectJ只是多了一些关键词罢了。

2.或者使用纯Java语言开发,然后使用AspectJ注解,简称@AspectJ。

AspectJ现在托管于Eclipse项目中,官方网站是:

下面我们以简单的程序来看下最简单的AspectJ语法:

@Aspect
public class AspectJTest {
private static final String TAG ="AspectJTest" ;
@Before("execution (* android.app.Activity.on**(..))")
public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable{
Log.e(TAG,joinPoint.getSignature().toString()+":beforecall"); }
}

@Aspect: 注解表明这是一个AspectJ文件,编译器在编译的时候,就会自动去解析,然后将代码注入到相应的JPonit中。

@Before: Advice,也就是具体的插入点。

execution: 处理JPoint的类型,例如call,execution

(* android.app.Activity.on**(..)):这个是最重要的表达式,第一个[]表示返回值,* 表示返回任意类型,后面的android.app.Activity.on**表示典型的包名路径,其中可用 * 来进行通配,几个 * 都没有区别。同时,这里可以通过 &&,||,! 来进行条件组合,例如:
@Before("execution ( android.app.Activity.onCreate(..)) || "
+ "execution (* android.app.Activity.onDestroy())")
public void onActivityMethodBefore(JoinPoint joinPoint) throws Throwable{
Log.e(TAG,joinPoint.getSignature().toString()+":beforecall");
}

从上面的例子我们可以看出,前面所用的AOP代码注入方法都是直接指定相应的pointcut进行切入。如果说此时我们又需要对某个函数进行代码切入,那么我们又必须重新写一个相应的pointcut,重新写个方法进行代码切入。那么,有没有更加方便的一种方法,可以让我们很方便的直接进行指定就能实现代码切入呢?通过自定义Pointcuts的方法可以实现上述所需功能。

自定义Pointcuts

自定义Pointcuts可以让我们更加精确的切入一个或多个指定的切入点。其具体操作步骤如下:

1.首先,我们需要自定义一个注解类

@Retention(RetentionPolicy.CLASS)
@Target({ElementType.CONSTRUCTOR,ElementType.METHOD})
public @interface AopLog {}

2.然后,创建一个切面文件,内部通过一个Pointcut来指定在带有我们上面自定义注解类AopLog注解的所有方法上进行拦截。

@Aspect
public class AopLogPointcut {
//在带有AopLog注解的方法进行切入(注:此处的 * *前面都要有一个空格)
@Pointcut("execution(@com.yn.aspectj.selfmakeaop.AopLog * *(..))")
public void logPointcut(){} //注意,这个函数必须要有实现,否则Java编译器会报错
@After("logPointcut()")
public void onLogPointcutAfter(JoinPoint joinPoint) throws Throwable{
Log.i("AOP","onLogPointcutAfter:"+joinPoint.getSignature());
}
}

3.最后,只需要在要进行切入的函数前加上@AopLog注解即可。

public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
testAopLog();
}
@AopLog
public void testAopLog() {
Log.i("AOP","in testAopLog");
}
}

掌握上述方法,则AOP基本编程便可以使用了。