深入理解java * 的两种实现方式(JDK/Cglib)
作者:PerKins.Zhu 时间:2023-11-26 13:29:52
什么是代理模式?
代理模式:在调用处不直接调用目标类进行操作,而是调用代理类,然后通过代理类来调用目标类进行操作。在代理类调用目标类的前后可以添加一些预处理和后处理操作来完成一些不属于目标类的功能。
为什么要使用代理模式?
通过代理模式可以实现对目标类调用的控制、在目标类调用前/后进行一些不属于目标类的操作,如:数据验证、预处理、后处理、异常处理等
什么是静态代理什么是 * ?
静态代理:代理类只能实现对”特定接口的实现类“进行代理
* :代理类可以实现对多种类的代理
jdk代理和cglib代理区别在哪里?
jdk * :代理所有“实现的有接口”的目标类
cglib * :代理任意一个目标类,但对final类和方法无法代理
不同点:jdk * 的目标类必须实现的有接口,因为在调用Proxy.newProxyInstance()的时候需要传入目标类的接口类。而cglib不做此限制。
下面看代码分析:
定义一个Person接口
package com.zpj.designMode.proxy;
//定义一个Person接口
public interface Person {
public void doWork();
}
添加一个实现类:MrLi
package com.zpj.designMode.proxy;
//添加一个实现类
public class MrLi implements Person {
@Override
public void doWork() {
System.out.println("-----doWork");
}
}
静态代理:
添加一个静态代理类Proxy
package com.zpj.designMode.proxy;
//静态代理,代理必须和目标类实现共同的接口
public class Proxy implements Person {
private Person person;// 被代理人
//这里的目标类型决定了该代理类只能代理实现了Person接口的实例,而不能接收其他类型参数,这也就是静态代理的局限性
public Proxy(Person person) {
this.person = person;
}
@Override
public void doWork() {
System.out.println("doSomething-----start");
person.doWork();
System.out.println("doSomething-----end");
}
}
静态代理测试程序:
package com.zpj.designMode.proxy;
public class Run {
public static void main(String[] args) {
MrLi li = new MrLi();
Proxy proxy = new Proxy(li);
//调用处直接调用代理进行目标方法的操作。
proxy.doWork();
}
}
JDK * :
添加一个代理JDKProxy,该代理实现InvocationHandler接口且覆写invoke方法。
package com.zpj.designMode.proxy.jdk;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/***
@author Perkins Zhu
@date 2017年3月13日 上午8:41:10
*/
public class JDKProxy implements InvocationHandler {
private Object person;// 被代理人
//这里的目标类型为Object,则可以接受任意一种参数作为被代理类,实现了 * 。但是要注意下面的newProxyInstance()中的参数
public Object getInstance(Object person) {
this.person = person;
//与cglib的区别在于这里构建代理对象的时候需要传入被代理对象的接口对象,第二个参数。而cglib不需要被代理对象实现任何接口即可
return Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("doSomething---------start");
method.invoke(person, args);
System.out.println("doSomething---------end");
return null;
}
}
JDK * 测试程序
package com.zpj.designMode.proxy.jdk;
import com.zpj.designMode.proxy.MrLi;
import com.zpj.designMode.proxy.Person;
/***
* @author Perkins Zhu
* @date 2017年3月13日 上午8:51:31
*/
public class Run {
public static void main(String[] args) {
Person person = (Person) new JDKProxy().getInstance(new MrLi());
//注意这里的person不是目标类person,而是代理类person:debug的时候显示null,有'$'标识符
person.doWork();
}
}
Cglib * :
添加一个CglibProxy代理,同时实现MethodInterceptor接口。
package com.zpj.designMode.proxy.cglib;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
/***
* @author Perkins Zhu
* @date 2017年3月13日 上午9:02:54
*/
public class CglibProxy implements MethodInterceptor {
private Object targetObject;
// 这里的目标类型为Object,则可以接受任意一种参数作为被代理类,实现了 *
public Object getInstance(Object target) {
this.targetObject = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
//注意该处代理的创建过程
Object proxyObj = enhancer.create();
return proxyObj;// 返回代理对象
}
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
Object obj = null;
System.out.println("doSomething---------start");
obj = method.invoke(targetObject, args);
System.out.println("doSomething---------end");
return obj;
}
}
Cglib * 测试程序
package com.zpj.designMode.proxy.cglib;
import com.zpj.designMode.proxy.MrLi;
import com.zpj.designMode.proxy.Person;
/***
@author Perkins Zhu
@date 2017年3月13日 上午9:07:38
*/
public class Run {
public static void main(String[] args) {
Person person = (Person)new CglibProxy().getInstance(new MrLi());
person.doWork();
}
}
仔细对比Proxy、CglibProxy和JDKProxy区分静态代理、JDK * 和Cglib * 的异同点!
来源:http://www.cnblogs.com/PerkinsZhu/p/6549211.html
标签:cglib,jdk,java
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
Android实现调用摄像头拍照与视频功能
2021-11-22 08:02:28
![](https://img.aspxhome.com/file/2023/2/137592_0s.jpg)
Java日常练习题,每天进步一点点(46)
2023-10-05 11:20:29
![](https://img.aspxhome.com/file/2023/0/116190_0s.png)
java private关键字用法实例
2022-01-16 10:08:06
Java开发中synchronized的定义及用法详解
2021-11-11 07:01:20
![](https://img.aspxhome.com/file/2023/6/62226_0s.png)
Mybatis接口式编程的原理
2023-11-27 22:16:05
Android实现极简打开摄像头
2022-09-10 15:56:51
![](https://img.aspxhome.com/file/2023/2/96802_0s.jpg)
c#与js随机数生成方法
2023-12-14 12:30:36
java 垃圾回收机制以及经典垃圾回收器详解
2022-07-06 05:16:08
![](https://img.aspxhome.com/file/2023/7/62067_0s.png)
Gradle:修改默认的Build配置文件名方式
2023-11-28 20:18:01
![](https://img.aspxhome.com/file/2023/7/60617_0s.png)
java并发编程专题(四)----浅谈(JUC)Lock锁
2021-11-30 00:20:23
![](https://img.aspxhome.com/file/2023/8/128858_0s.png)
Android 无障碍服务 performAction 调用过程分析
2023-05-03 12:06:17
Java并发问题之乐观锁与悲观锁
2022-07-13 19:59:05
![](https://img.aspxhome.com/file/2023/1/84201_0s.png)
基于C#实现的仿windows左侧伸缩菜单效果
2023-01-22 18:44:54
使用Java8 Stream流的skip + limit实现批处理的方法
2023-11-29 06:17:39
Android实现雷达View效果的示例代码
2022-09-27 11:39:52
![](https://img.aspxhome.com/file/2023/1/139381_0s.jpg)
IDEA设置生成带注释的getter和setter的图文教程
2023-06-28 00:49:36
![](https://img.aspxhome.com/file/2023/3/132223_0s.png)
Springboot初始化项目并完成登入注册的全过程
2023-07-31 15:09:09
![](https://img.aspxhome.com/file/2023/8/57948_0s.png)
java生成随机字符串的两种方法
2022-09-09 22:49:26
java http token请求代码实例
2022-09-28 18:23:19
Java实现的简单网页截屏功能示例
2021-05-25 13:50:35
![](https://img.aspxhome.com/file/2023/9/88219_0s.png)