1
package sinosoft.dj.aop.proxyaop;
2
3
import java.lang.reflect.Method;
4
5
public interface IOperation {
6
/**
7
* 方法执行之前的操作
8
* @param method
9
*/
10
void start(Method method);
11
/**
12
* 方法执行之后的操作
13
* @param method
14
*/
15
void end(Method method);
16
}
17

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

我们去写一个实现上面接口的类.我们把作他真正的操作者,如下面是日志操作者的一个类:
LoggerOperation.java

















然后我们要改一下代理对象DynaProxyHello中的代码.如下:
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
*/
11
private Object proxy;
12
/**
13
* 要处理的对象(也就是我们要在方法的前后加上业务逻辑的对象,如例子中的Hello)
14
*/
15
private Object delegate;
16
17
/**
18
* 动态生成方法被处理过后的对象 (写法固定)
19
*
20
* @param delegate
21
* @param proxy
22
* @return
23
*/
24
public Object bind(Object delegate,Object proxy) {
25
26
this.proxy = proxy;
27
this.delegate = delegate;
28
return Proxy.newProxyInstance(
29
this.delegate.getClass().getClassLoader(), this.delegate
30
.getClass().getInterfaces(), this);
31
}
32
/**
33
* 要处理的对象中的每个方法会被此方法送去JVM调用,也就是说,要处理的对象的方法只能通过此方法调用
34
* 此方法是动态的,不是手动调用的
35
*/
36
public Object invoke(Object proxy, Method method, Object[] args)
37
throws Throwable {
38
Object result = null;
39
try {
40
//反射得到操作者的实例
41
Class clazz = this.proxy.getClass();
42
//反射得到操作者的Start方法
43
Method start = clazz.getDeclaredMethod("start",
44
new Class[] { Method.class });
45
//反射执行start方法
46
start.invoke(this.proxy, new Object[] { method });
47
//执行要处理对象的原本方法
48
result = method.invoke(this.delegate, args);
49
// 反射得到操作者的end方法
50
Method end = clazz.getDeclaredMethod("end",
51
new Class[] { Method.class });
52
// 反射执行end方法
53
end.invoke(this.proxy, new Object[] { method });
54
55
} catch (Exception e) {
56
e.printStackTrace();
57
}
58
return result;
59
}
60
61
}
62

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

然后我们把Test.java中的代码改一下.测试一下:











如果你想在每个方法之前加上日志记录,而不在方法后加上日志记录.你就把LoggerOperation类改成如下:
1
package sinosoft.dj.aop.proxyaop;
2
3
import java.lang.reflect.Method;
4
5
public class LoggerOperation implements IOperation {
6
7
public void end(Method method) {
8
//Logger.logging(Level.DEBUGE, method.getName() + " Method end
.");
9
}
10
11
public void start(Method method) {
12
Logger.logging(Level.INFO, method.getName() + " Method Start!");
13
}
14
15
}
16

2

3

4

5

6

7

8


9

10

11

12

13

14

15

16

运行一下.你就会发现,每个方法之后没有记录日志了. 这样,我们就把代理者和操作者解藕了!
下面留一个问题给大家,如果我们不想让所有方法都被日志记录,我们应该怎么去解藕呢.?
我的想法是在代理对象的public Object invoke(Object proxy, Method method, Object[] args)方法里面加上个if(),对传进来的method的名字进行判断,判断的条件存在XML里面.这样我们就可以配置文件时行解藕了.如果有兴趣的朋友可以把操作者,被代理者,都通过配置文件进行配置 ,那么就可以写一个简单的SpringAOP框架了.