其实AOP的意思就是面向切面编程.
OO注重的是我们解决问题的方法(封装成Method),而AOP注重的是许多解决解决问题的方法中的共同点,是对OO思想的一种补充!
还是拿人家经常举的一个例子讲解一下吧:
比如说,我们现在要开发的一个应用里面有很多的业务方法,但是,我们现在要对这个方法的执行做全面监控,或部分监控.也许我们就会在要一些方法前去加上一条日志记录,
我们写个例子看看我们最简单的解决方案
我们先写一个接口IHello.java代码如下:
1
package
sinosoft.dj.aop.staticaop;
2

3

public
interface
IHello
{
4

/** */
/**
5
* 假设这是一个业务方法
6
*
@param
name
7
*/
8
void
sayHello(String name);
9
}
10
里面有个方法,用于输入"Hello" 加传进来的姓名;我们去写个类实现IHello接口
package
sinosoft.dj.aop.staticaop;


public
class
Hello
implements
IHello
{


public
void
sayHello(String name)
{
System.out.println(
"
Hello
"
+
name);
}

}
现在我们要为这个业务方法加上日志记录的业务,我们在不改变原代码的情况下,我们会去怎么做呢?也许,你会去写一个类去实现IHello接口,并依赖Hello这个类.代码如下:
1
package
sinosoft.dj.aop.staticaop;
2

3

public
class
HelloProxy
implements
IHello
{
4
private
IHello hello;
5

6

public
HelloProxy(IHello hello)
{
7
this
.hello
=
hello;
8
}
9

10

public
void
sayHello(String name)
{
11
Logger.logging(Level.DEBUGE,
"
sayHello method start
.
"
);
12
hello.sayHello(name);
13
Logger.logging(Level.INFO,
"
sayHello method end!
"
);
14

15
}
16

17
}
18
其中.Logger类和Level枚举代码如下:
Logger.java
1
package
sinosoft.dj.aop.staticaop;
2

3
import
java.util.Date;
4

5

public
class
Logger
{
6

/** */
/**
7
* 根据等级记录日志
8
*
@param
level
9
*
@param
context
10
*/
11

public
static
void
logging(Level level, String context)
{
12

if
(level.equals(Level.INFO))
{
13
System.out.println(
new
Date().toLocaleString()
+
"
"
+
context);
14
}
15

if
(level.equals(Level.DEBUGE))
{
16
System.err.println(
new
Date()
+
"
"
+
context);
17
}
18
}
19

20
}
21
Level.java
1
package
sinosoft.dj.aop.staticaop;
2

3

public
enum
Level
{
4
INFO,DEBUGE;
5
}
6
那我们去写个测试类看看,代码如下:
Test.java
1
package
sinosoft.dj.aop.staticaop;
2

3

public
class
Test
{
4

public
static
void
main(String[] args)
{
5
IHello hello
=
new
HelloProxy(
new
Hello());
6
hello.sayHello(
"
Doublej
"
);
7
}
8
}
9
运行以上代码我们可以得到下面结果:
Tue Mar
04
20
:
57
:
12
CST
2008
sayHello method start
.
Hello Doublej
2008
-
3
-
4
20
:
57
:
12
sayHello method end!
从上面的代码我们可以看出,hello对象是被HelloProxy这个所谓的代理态所创建的.这样,如果我们以后要把日志记录的功能去掉.那我们只要把得到hello对象的代码改成以下:
1
package
sinosoft.dj.aop.staticaop;
2

3

public
class
Test
{
4

public
static
void
main(String[] args)
{
5
IHello hello
=
new
Hello();
6
hello.sayHello(
"
Doublej
"
);
7
}
8
}
9
上面代码,可以说是AOP最简单的实现!
但是我们会发现一个问题,如果我们像Hello这样的类很多,那么,我们是不是要去写很多个HelloProxy这样的类呢.没错,是的.其实也是一种很麻烦的事.在jdk1.3以后.jdk跟我们提供了一个API
java.lang.reflect.InvocationHandler的类.
这个类可以让我们在JVM调用某个类的方法时动态的为些方法做些什么事.让我们把以上的代码改一下来看看效果.
同样,我们写一个IHello的接口和一个Hello的实现类.在接口中.我们定义两个方法;代码如下
:
IHello.java
1
package
sinosoft.dj.aop.proxyaop;
2

3

public
interface
IHello
{
4

/** */
/**
5
* 业务处理A方法
6
*
@param
name
7
*/
8
void
sayHello(String name);
9

/** */
/**
10
* 业务处理B方法
11
*
@param
name
12
*/
13
void
sayGoogBye(String name);
14
}
15
Hello.java
1
package
sinosoft.dj.aop.proxyaop;
2

3

public
class
Hello
implements
IHello
{
4

5

public
void
sayHello(String name)
{
6
System.out.println(
"
Hello
"
+
name);
7
}
8

public
void
sayGoogBye(String name)
{
9
System.out.println(name
+
"
GoodBye!
"
);
10
}
11
}
12
我们一样的去写一个代理类.只不过.让这个类去实现java.lang.reflect.InvocationHandler接口,代码如下:
1
package
sinosoft.dj.aop.proxyaop;
2

3
import
java.lang.reflect.InvocationHandler;
4
import
java.lang.reflect.Method;
5
import
java.lang.reflect.Proxy;
6

7

public
class
DynaProxyHello
implements
InvocationHandler
{
8

9

/** */
/**
10
* 要处理的对象(也就是我们要在方法的前后加上业务逻辑的对象,如例子中的Hello)
11
*/
12
private
Object delegate;
13

14

/** */
/**
15
* 动态生成方法被处理过后的对象 (写法固定)
16
*
17
*
@param
delegate
18
*
@param
proxy
19
*
@return
20
*/
21

public
Object bind(Object delegate)
{
22
this
.delegate
=
delegate;
23
return
Proxy.newProxyInstance(
24
this
.delegate.getClass().getClassLoader(),
this
.delegate
25
.getClass().getInterfaces(),
this
);
26
}
27

/** */
/**
28
* 要处理的对象中的每个方法会被此方法送去JVM调用,也就是说,要处理的对象的方法只能通过此方法调用
29
* 此方法是动态的,不是手动调用的
30
*/
31
public
Object invoke(Object proxy, Method method, Object[] args)
32

throws
Throwable
{
33
Object result
=
null
;
34

try
{
35
//
执行原来的方法之前记录日志
36
Logger.logging(Level.DEBUGE, method.getName()
+
"
Method end
.
"
);
37
38
//
JVM通过这条语句执行原来的方法(反射机制)
39
result
=
method.invoke(
this
.delegate, args);
40
//
执行原来的方法之后记录日志
41
Logger.logging(Level.INFO, method.getName()
+
"
Method Start!
"
);
42

}
catch
(Exception e)
{
43
e.printStackTrace();
44
}
45
//
返回方法返回值给调用者
46
return
result;
47
}
48

49
}
50
上面类中出现的Logger类和Level枚举还是和上一上例子的实现是一样的.这里就不贴出代码了.
让我们写一个Test类去测试一下.代码如下:
Test.java
1
package
sinosoft.dj.aop.proxyaop;
2

3

public
class
Test
{
4

public
static
void
main(String[] args)
{
5
IHello hello
=
(IHello)
new
DynaProxyHello().bind(
new
Hello());
6
hello.sayGoogBye(
"
Double J
"
);
7
hello.sayHello(
"
Double J
"
);
8
9
}
10
}
11
http://www.blogjava.net/DoubleJ/archive/2008/03/04/183796.html
运行输出的结果如下: