一、基本概念
AspectJ 是一种面向切面程序设计的基于 Java 的实现。它向 Java 中加入了连接点(Join Point)这个新概念,其实它也只是现存的一个 Java概念的名称而已。它向 Java 语言中加入少许新结构:切点(pointcut)、通知(Advice)、类型间声明(Inter-type declaration)和方面(Aspect)。切点和通知动态地影响程序流程,类型间声明则是静态的影响程序的类等级结构,而切面则是对所有这些新结构的封装。
基于切面、连接点、切点、通知的概念如下:
- Aspect: Aspect 声明类似于 Java 中的类声明,在 Aspect 中会包含着一些 Pointcut 以及相应的 Advice。
- Joint point:表示在程序中明确定义的点,典型的包括方法调用,对类成员的访问以及异常处理程序块的执行等等,它自身还可以嵌套其它 joint point。
- Pointcut:表示一组 joint point,这些 joint point 或是通过逻辑关系组合起来,或是通过通配、正则表达式等方式集中起来,它定义了相应的 Advice 将要发生的地方。
- Advice:Advice 定义了在 pointcut 里面定义的程序点具体要做的操作,它通过 before、after 和 around 来区别是在每个 joint point 之前、之后还是代替执行的代码。
连接点是程序流中适当的一点。切点收集特定的连接点集合和在这些点中的值。一个通知是当一个连接点到达时执行的代码,这些都是 AspectJ的动态部分。其实连接点就好比是程序中的一条一条的语句,而切点就是特定一条语句处设置的一个断点,它收集了断点处程序栈的信息,而通知就是在这个断点前后想要加入的程序代。AspectJ 中也有许多不同种类的类型间声明, 这就允许程序员修改程序的静态结构、 名称、 类的成员以及类之间的关系。AspectJ 中的方面是横切关注点的模块单元。它们的行为与 Java语言中的类很象,但是方面还封装了切点、通知以及类型间声明
二、如何基于AJDT: AspectJ Development Tools进行AOP程序的开发?
- 按照ajdt网站上的说明,在eclipse中安装ajdt插件http://www.eclipse.org/ajdt/ ajdt - http://download.eclipse.org/tools/ajdt/43/update
- 创建AspectJ Project项目
- 进行代码开发(本文提供一个简单的示例)
三、简单示例:
package aop.test;
public interface FigureElement {
public void setXY(int x,int y);
public void draw();
}
package aop.test;
public class Point implements FigureElement {
public int x;
private int y;
public int getX() {
return x;
}
public String setX(int x) {
System.out.println("设置x值:x="+x);
this.x = x;
return "返回值是x="+x;
}
public int getY() {
return y;
}
public void setY(int y) {
this.y = y;
}
public Point(){
}
public Point(int x,int y){
this.x=x;
this.y=y;
}
@Override
public void setXY(int x,int y) {
this.x=x;
this.y=y;
System.out.println("Point setXY: x="+x+",y="+y);
}
@Override
public void draw() {
System.out.println("Point draw");
}
@Override
public String toString(){
return "Point: x="+x+",y="+y;
}
}
package aop.test;
public aspect TestMain {
/**
* 不带参数的pointcut
*/
pointcut psimple() : call(void Point.*(int));
before() : psimple() {
System.out.println("psimpe执行前调用");
}
after() returning() : psimple() {
System.out.println("psimple执行后调用");
}
/**
* 带参数的pointcut
*
* @param i
*/
pointcut psetX(int i) : call(String Point.setX(int)) && args(i);
before(int i) : psetX(i) {
System.out.println("psetX执行前调用" + i);
}
after(int i) returning(String sReturn) : psetX(i) {
System.out.println("psetX执行后调用" + i + ":" + sReturn);
}
/**
* 带多个参数的pointcut
*
* @param fe
* @param x
* @param y
*/
pointcut psetXY(FigureElement fe, int x, int y) :
call(void FigureElement.setXY(int,int))
&& target(fe)
&& args(x,y);
// before(FigureElement fe, int x, int y) : psetXY(fe,x,y) {
// System.out.println("psetXY before" + fe + " move to x=" + x + ",y=" + y);
// }
after(FigureElement fe, int x, int y) returning() : psetXY(fe,x,y) {
System.out.println("psetXY after" + fe + " move to x=" + x + ",y=" + y);
}
before(FigureElement fe, int x, int y) :
call(void FigureElement.setXY(int,int))
&& target(fe)
&& args(x,y) {
System.out.println("psetXY before" + fe + " move to x=" + x + ",y=" + y);
}
public static void main(String[] args) {
Point p1 = new Point();
p1.setY(2);
p1.setX(2);
Point p2 = new Point(2, 2);
p2.setXY(2, 3);
}
}