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
package sinosoft.dj.aop.proxyaop;2

3
import java.lang.reflect.Method;4

5
public interface IOperation {6
/**7
* 方法执行之前的操作8
* @param method9
*/10
void start(Method method);11
/**12
* 方法执行之后的操作13
* @param method14
*/15
void end(Method method);16
}17
我们去写一个实现上面接口的类.我们把作他真正的操作者,如下面是日志操作者的一个类:
LoggerOperation.java
package sinosoft.dj.aop.proxyaop;
import java.lang.reflect.Method;
public class LoggerOperation implements IOperation {
public void end(Method method) {
Logger.logging(Level.DEBUGE, method.getName() + " Method end
.");
}
public void start(Method method) {
Logger.logging(Level.INFO, method.getName() + " Method Start!");
}
}
然后我们要改一下代理对象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
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 delegate21
* @param proxy22
* @return23
*/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.delegate30
.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
然后我们把Test.java中的代码改一下.测试一下:
package sinosoft.dj.aop.proxyaop;
public class Test {
public static void main(String[] args) {
IHello hello = (IHello)new DynaProxyHello().bind(new Hello(),new LoggerOperation());
hello.sayGoogBye("Double J");
hello.sayHello("Double J");
}
}
如果你想在每个方法之前加上日志记录,而不在方法后加上日志记录.你就把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
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
运行一下.你就会发现,每个方法之后没有记录日志了. 这样,我们就把代理者和操作者解藕了!
下面留一个问题给大家,如果我们不想让所有方法都被日志记录,我们应该怎么去解藕呢.?
我的想法是在代理对象的public Object invoke(Object proxy, Method method, Object[] args)方法里面加上个if(),对传进来的method的名字进行判断,判断的条件存在XML里面.这样我们就可以配置文件时行解藕了.如果有兴趣的朋友可以把操作者,被代理者,都通过配置文件进行配置 ,那么就可以写一个简单的SpringAOP框架了.
本文介绍了一个基于Java的AOP(面向切面编程)实践案例,通过自定义代理类实现了方法执行前后的日志记录功能,并展示了如何通过动态代理实现解耦。
1452

被折叠的 条评论
为什么被折叠?



