java 动态代理模式(jdk和cglib)

 1 package proxy.dynamicproxy;
 2 
 3 import java.lang.reflect.InvocationHandler;
 4 import java.lang.reflect.Method;
 5 import java.lang.reflect.Proxy;
 6 
 7 
 8 /**
 9  * 需要实现InvocationHandler接口,内部维护一个实际类实例
10  *
11  */
12 public class JdkProxyHandler implements InvocationHandler {
13 
14     private Object realObject;
15 
16     public Object proxy(Object realObject){
17         this.realObject = realObject;
18         return Proxy.newProxyInstance(this.realObject.getClass().getClassLoader(),
19                 this.realObject.getClass().getInterfaces(), this);
20     }
21 
22     /**
23      *
24      * @param proxy 动态生成的代理类实例
25      * @param method 方法实例
26      * @param args 方法参数
27      * @return
28      * @throws Throwable
29      */
30     @Override
31     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
32         System.out.println("我是代理人:大明星唱歌开始前,我先宣传一下:巴拉巴拉。。。");
33 
34         // 调用实际类的方法,并传入参数,内部是反射机制
35         Object ret = method.invoke(this.realObject, args);
36 
37         System.out.println("我是代理人:大明星唱歌完毕了,我来总结一下:巴拉巴拉。。。");
38 
39         return ret;
40     }
41 }
 1 package proxy.dynamicproxy;
 2 
 3 import net.sf.cglib.proxy.Enhancer;
 4 import net.sf.cglib.proxy.MethodInterceptor;
 5 import net.sf.cglib.proxy.MethodProxy;
 6 
 7 import java.lang.reflect.Method;
 8 
 9 /**
10  * 需要实现MethodInterceptor接口
11  * cglib相关依赖:
12  * ant-1.6.2.jar
13  * asm-3.1.jar
14  * asm-util-3.1.jar
15  * cglib-2.2.2.jar
16  */
17 public class CglibProxyHandler implements MethodInterceptor {
18 
19     public Object proxy(Object realObject){
20 
21         // 使用字节码增强器 四个固定步骤:
22         // 1、new字节码增强器
23         // 2、设置当前类实例为回调
24         // 3、将实际类实例设置为父类
25         // 4、创建一个代理类
26         Enhancer enhancer = new Enhancer();
27         enhancer.setCallback(this);
28         enhancer.setSuperclass(realObject.getClass());
29         // 这里会生成代理类、代理类的FastClass辅助类、实际类的FastClass辅助类
30         // 辅助类为代理类和实际类的每个方法生成一个唯一的id
31         // 用于在调用intercept方法时,通过唯一id就可以调用对应的方法
32         // 不再走反射机制,提高性能
33         return enhancer.create();
34     }
35 
36     /**
37      *
38      * @param o 代理类的实例
39      * @param method 方法实例
40      * @param objects 方法参数
41      * @param methodProxy 方法代理
42      * @return
43      * @throws Throwable
44      */
45     @Override
46     public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
47 
48         System.out.println("我是代理人:大明星唱歌开始前,我先宣传一下:巴拉巴拉。。。");
49 
50         // 这里如果使用 method.invoke方法,就等同于走了反射机制去调用方法,性能不高
51         // 而且还需要另外维护实际类实例
52         // Object ret = method.invoke(this.realObject, objects);
53 
54         Object ret = methodProxy.invokeSuper(o, objects);
55 
56         System.out.println("我是代理人:大明星唱歌完毕了,我来总结一下:巴拉巴拉。。。");
57 
58         return ret;
59     }
60 }
 1 package proxy.dynamicproxy;
 2 
 3 import net.sf.cglib.core.DebuggingClassWriter;
 4 import proxy.staticproxy.IStar;
 5 import proxy.staticproxy.RealStar;
 6 
 7 public class Test {
 8     public static void main(String[] args) {
 9 
10         /**
11          * 还是以“代理人”和“大明星”为例
12          * jdk动态代理:适用于大明星实现某接口的情况,且只能用于实现接口的情况
13          *      不能用于未实现任何接口的类,因为生成的动态代理类要继承自Proxy、同时实现大明星接口。
14          * cglib动态代理:适用于任何类。它是采用动态代理类直接继承大明星类的方式,将大明星当作父类
15          *      覆写大明星类的所有方法(除final修饰的方法,wait方法,notify方法)
16          *
17          * 优缺点:
18          * jdk方式,只能针对接口,底层直接写字节码的方式生成代理类,所以生成代理类速度快
19          *      但是代理类执行方法时,通过反射的方式去执行,速度不如cglib方式
20          * cglib方式,可以适用于任何类,底层使用ASM框架生成字节码,因为采用FastClass机制
21          *      在生成代理类的同时还要生成代理类和大明星类的对应FastClass类(辅助类)
22          *      这两个辅助类的作用是:对应FastClass辅助类会为代理类和大明星类的每一个方法
23          *      (除final修饰的方法,wait方法,notify方法)生成唯一id,这样在后面的调用方法时
24          *      不再通过反射去执行逻辑,而是直接根据id找到对应的方法去执行,提高性能,但相对的,生成字节码速度较慢
25          */
26         boolean isUseJdkProxy = false;
27 
28         if (isUseJdkProxy) {
29             System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
30 
31             IStar star = (IStar) new JdkProxyHandler().proxy(new RealStar());
32             star.sing();
33         }
34         else {
35 
36             System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,
37                     "C:\\Users\\Administrator\\IdeaProjects\\untitled\\cglibClass");
38 
39             IStar star = (IStar) new CglibProxyHandler().proxy(new RealStar());
40 
41             // 这一步里面的具体流程:
42             // 动态代理类的sing方法 --> CglibProxyHandler的intercept方法
43             //  --> 实际类执行前的行为代理 --> MethodProxy.invokeSuper方法
44             //  --> 根据唯一id在FastClass里找到对应的实际方法
45             //  --> 代理类的FastClass内部:让代理类调用实际方法
46             //  --> 代理类的实际方法内部一般就是直接调用父类(被代理类)的方法
47             //  --> 返回父类方法的返回值
48             star.sing();
49 
50         }
51     }
52 }

 

上一篇:Linux下用于查看系统当前登录用户信息 w命令


下一篇:Spring-6.1、Java三种代理模式:静态代理、动态代理和cglib代理