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)的更多相关文章

  1. [转]java动态代理(JDK和cglib)

    转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html java动态代理(JDK和cglib) JAVA的动态代理 代理模式 代理 ...

  2. java动态代理(JDK和cglib)

    转:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是代 ...

  3. 《转》JAVA动态代理(JDK和CGLIB)

    该文章转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的 ...

  4. 《转》java动态代理(JDK和cglib)

    该文章转自:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的 ...

  5. java动态代理(JDK和cglib实现对比)

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt214 JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特 ...

  6. Spring学习(四)—— java动态代理(JDK和cglib)

    JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他 的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托 ...

  7. java动态代理(JDK和cglib)(转载)

    原文地址:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html 高亮部分是我的理解. JAVA的动态代理 代理模式 代理模式是常用的j ...

  8. java动态代理(JDK和CGLIB)笔记

    动态代理:为一堆interface或类的实现提供统一的执行通道,从含义上就像局域网电脑通过代理上网一样,走统一的通道,代理控制通道,自然可以在通道里加上自定义实现,例如像AOP切面,日志等. JDK的 ...

  9. Java 的静态代理 动态代理(JDK和cglib)

    转载:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是 ...

  10. Java中的动态代理(jdk和cglib)

    JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类 ...

随机推荐

  1. Java定义三个点Object...

    从Java 5开始,Java语言对方法参数支持一种新写法,叫 可变长度参数列表,其语法就是类型后跟...,表示此处接受的参数为0到多个Object类型的对象,或者是一个Object[]. public ...

  2. MySQL学习基础知识1

    什么是数据库? 数据库就是存储数据的仓库. 存储方式: 变量 无法永久存储 文件处理,可以永久存储,弊端:文件只能在自己的计算机读写,无法被分享(局域网除外) 数据库分类: 1.关系型数据库 提供某种 ...

  3. 商誉专题RN及H5项目总结

    React(基础框架): React 是基础框架,是一套基础设计实现理念,开发者不能直接使用它来开发移动应用或者网页. React.js(web网页开发):在React框架之上,发展出了React.j ...

  4. python三大器(装饰器/生成器/迭代器)

    1装饰器 1.1基本结构 def 外层函数(参数): def 内层函数(*args,**kwargs); return 参数(*args,**kwargs) return 内层函数 @外层函数 def ...

  5. python@wraps实现原理

    @wraps作用 python中的装饰器装饰过的函数其实就不是函数本身了,我们可以看看下面的例子 import time def timmer(func): """tim ...

  6. 关于ajax 进行post提交 json数据到controller

    首选需要参考的两个博客: www.cnblogs.com/Benjamin/archive/2013/09/11/3314576.html http://www.cnblogs.com/quanyon ...

  7. Django框架基础之Form组件

    服务端假设所有用户提交的数据都是不可信任的,所以Django框架内置了form组件来验证用户提交的信息 form组件的2大功能:       1 验证(显示错误信息)       2 保留用户上次输入 ...

  8. consul实现分布式锁

    分布式一致性问题: 分布式的CAP理论告诉我们"任何一个分布式系统都无法同时满足一致性(Consistency).可用性(Availability)和分区容错性(Partition tole ...

  9. ccf 201503-5 最小花费 这题交上去只有10分嗨!求大佬的题解啊

    问题描述 C国共有n个城市.有n-1条双向道路,每条道路连接两个城市,任意两个城市之间能互相到达.小R来到C国旅行,他共规划了m条旅行的路线,第i条旅行路线的起点是si,终点是ti.在旅行过程中,小R ...

  10. Java SE之[静态成员/类成员]与[非静态成员/实例成员]【static】

    定义 静态成员:又称类成员,使用static修饰符的方法和变量: 非静态成员:又称实例成员,未使用static修饰符的方法和变量. 结论 注:jdk1.8 测试源码 public class Main ...