java 动态代理模式(jdk和cglib)
package proxy.dynamicproxy; import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy; /**
* 需要实现InvocationHandler接口,内部维护一个实际类实例
*
*/
public class JdkProxyHandler implements InvocationHandler { private Object realObject; public Object proxy(Object realObject){
this.realObject = realObject;
return Proxy.newProxyInstance(this.realObject.getClass().getClassLoader(),
this.realObject.getClass().getInterfaces(), this);
} /**
*
* @param proxy 动态生成的代理类实例
* @param method 方法实例
* @param args 方法参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("我是代理人:大明星唱歌开始前,我先宣传一下:巴拉巴拉。。。"); // 调用实际类的方法,并传入参数,内部是反射机制
Object ret = method.invoke(this.realObject, args); System.out.println("我是代理人:大明星唱歌完毕了,我来总结一下:巴拉巴拉。。。"); return ret;
}
}
package proxy.dynamicproxy; import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; /**
* 需要实现MethodInterceptor接口
* cglib相关依赖:
* ant-1.6.2.jar
* asm-3.1.jar
* asm-util-3.1.jar
* cglib-2.2.2.jar
*/
public class CglibProxyHandler implements MethodInterceptor { public Object proxy(Object realObject){ // 使用字节码增强器 四个固定步骤:
// 1、new字节码增强器
// 2、设置当前类实例为回调
// 3、将实际类实例设置为父类
// 4、创建一个代理类
Enhancer enhancer = new Enhancer();
enhancer.setCallback(this);
enhancer.setSuperclass(realObject.getClass());
// 这里会生成代理类、代理类的FastClass辅助类、实际类的FastClass辅助类
// 辅助类为代理类和实际类的每个方法生成一个唯一的id
// 用于在调用intercept方法时,通过唯一id就可以调用对应的方法
// 不再走反射机制,提高性能
return enhancer.create();
} /**
*
* @param o 代理类的实例
* @param method 方法实例
* @param objects 方法参数
* @param methodProxy 方法代理
* @return
* @throws Throwable
*/
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("我是代理人:大明星唱歌开始前,我先宣传一下:巴拉巴拉。。。"); // 这里如果使用 method.invoke方法,就等同于走了反射机制去调用方法,性能不高
// 而且还需要另外维护实际类实例
// Object ret = method.invoke(this.realObject, objects); Object ret = methodProxy.invokeSuper(o, objects); System.out.println("我是代理人:大明星唱歌完毕了,我来总结一下:巴拉巴拉。。。"); return ret;
}
}
package proxy.dynamicproxy; import net.sf.cglib.core.DebuggingClassWriter;
import proxy.staticproxy.IStar;
import proxy.staticproxy.RealStar; public class Test {
public static void main(String[] args) { /**
* 还是以“代理人”和“大明星”为例
* jdk动态代理:适用于大明星实现某接口的情况,且只能用于实现接口的情况
* 不能用于未实现任何接口的类,因为生成的动态代理类要继承自Proxy、同时实现大明星接口。
* cglib动态代理:适用于任何类。它是采用动态代理类直接继承大明星类的方式,将大明星当作父类
* 覆写大明星类的所有方法(除final修饰的方法,wait方法,notify方法)
*
* 优缺点:
* jdk方式,只能针对接口,底层直接写字节码的方式生成代理类,所以生成代理类速度快
* 但是代理类执行方法时,通过反射的方式去执行,速度不如cglib方式
* cglib方式,可以适用于任何类,底层使用ASM框架生成字节码,因为采用FastClass机制
* 在生成代理类的同时还要生成代理类和大明星类的对应FastClass类(辅助类)
* 这两个辅助类的作用是:对应FastClass辅助类会为代理类和大明星类的每一个方法
* (除final修饰的方法,wait方法,notify方法)生成唯一id,这样在后面的调用方法时
* 不再通过反射去执行逻辑,而是直接根据id找到对应的方法去执行,提高性能,但相对的,生成字节码速度较慢
*/
boolean isUseJdkProxy = false; if (isUseJdkProxy) {
System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true"); IStar star = (IStar) new JdkProxyHandler().proxy(new RealStar());
star.sing();
}
else { System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,
"C:\\Users\\Administrator\\IdeaProjects\\untitled\\cglibClass"); IStar star = (IStar) new CglibProxyHandler().proxy(new RealStar()); // 这一步里面的具体流程:
// 动态代理类的sing方法 --> CglibProxyHandler的intercept方法
// --> 实际类执行前的行为代理 --> MethodProxy.invokeSuper方法
// --> 根据唯一id在FastClass里找到对应的实际方法
// --> 代理类的FastClass内部:让代理类调用实际方法
// --> 代理类的实际方法内部一般就是直接调用父类(被代理类)的方法
// --> 返回父类方法的返回值
star.sing(); }
}
}
java 动态代理模式(jdk和cglib)的更多相关文章
- [转]java动态代理(JDK和cglib)
转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html java动态代理(JDK和cglib) JAVA的动态代理 代理模式 代理 ...
- java动态代理(JDK和cglib)
转:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是代 ...
- 《转》JAVA动态代理(JDK和CGLIB)
该文章转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的 ...
- 《转》java动态代理(JDK和cglib)
该文章转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的 ...
- java动态代理(JDK和cglib实现对比)
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt214 JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特 ...
- Spring学习(四)—— java动态代理(JDK和cglib)
JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他 的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托 ...
- java动态代理(JDK和cglib)(转载)
原文地址:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html 高亮部分是我的理解. JAVA的动态代理 代理模式 代理模式是常用的j ...
- java动态代理(JDK和CGLIB)笔记
动态代理:为一堆interface或类的实现提供统一的执行通道,从含义上就像局域网电脑通过代理上网一样,走统一的通道,代理控制通道,自然可以在通道里加上自定义实现,例如像AOP切面,日志等. JDK的 ...
- Java 的静态代理 动态代理(JDK和cglib)
转载:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是 ...
- Java中的动态代理(jdk和cglib)
JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类 ...
随机推荐
- UOJ14 UER #1 DZY Loves Graph(最小生成树+并查集)
显然可以用可持久化并查集实现.考虑更简单的做法.如果没有撤销操作,用带撤销并查集暴力模拟即可,复杂度显然可以均摊.加上撤销操作,删除操作的复杂度不再能均摊,但注意到我们在删除时就可以知道他会不会被撤销 ...
- Nginx从入门到实践(二)
静态资源web服务 静态资源类型 CDN CDN的基本原理是广泛采用各种缓存服务器,将这些缓存服务器分布到用户访问相对集中的地区或网络中,在用户访问网站时,利用全局负载技术将用户的访问指向距离最近的工 ...
- 函数中的this与argument对象,以及argument中的callee与caller属性
相关阅读:https://segmentfault.com/a/1190000015438195 相关阅读: https://zhuanlan.zhihu.com/p/23804247 相关阅读: h ...
- CENTOS手动安装修复python ,YUM
CENTOS手动安装修复YUM
CENTOS手动安装修复YUM 2019年3月8日 杨宇 Comments 0 Comment 目录 [hide] 一.问题场景 二.手动修复 2.1 下载 rpm 包 2.2 安装 rpm 包 ...
- Re.多项式除法/取模
前言 emmm又是暂无 前置 多项式求逆 多项式除法/取模目的 还是跟之前一样顾名思义] 给定一个多项式F(x),请求出多项式Q(x)和R(x),满足F(x)=Q(x)∗G(x)+R(x),R项数小于 ...
- Python成绩
# -*- coding: utf-8 -*- """ Spyder Editor This is a temporary script file. "&quo ...
- 公共DNS推荐及dns测速
DNS在平时上网中扮演重要角色,如果不注意DNS的话,可能会导致网速慢.弹窗广告.网址打不开.打开不是自己想要的网站.劫持等一系列问题.针对DNS的问题,今天我们就来总结一下,看看哪个DNS服务器最好 ...
- 关于ajax的与后台Controller的交互 后台拿不到值
话不多说 上代码 这是前段js的代码 传的两个参数 cLassid 和 userid $.ajax({ type:"post", url:"../ ...
- 爬虫框架Scrapy 之(二) --- scrapy文件
框架简介 核心部分: 引擎.下载器.调度器 自定义部分: spider(自己建的爬虫文件).管道(pipelines.py) 目录结构 firstSpider firstSpider spiders ...
- Technocup 2019 - Elimination Round 2
http://codeforces.com/contest/1031 (如果感觉一道题对于自己是有难度的,不要后退,懂0%的时候敲一遍,边敲边想,懂30%的时候敲一遍,边敲边想,懂60%的时候敲一遍, ...