一、基本概念
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插件 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); }}