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. UOJ14 UER #1 DZY Loves Graph(最小生成树+并查集)

    显然可以用可持久化并查集实现.考虑更简单的做法.如果没有撤销操作,用带撤销并查集暴力模拟即可,复杂度显然可以均摊.加上撤销操作,删除操作的复杂度不再能均摊,但注意到我们在删除时就可以知道他会不会被撤销 ...

  2. Nginx从入门到实践(二)

    静态资源web服务 静态资源类型 CDN CDN的基本原理是广泛采用各种缓存服务器,将这些缓存服务器分布到用户访问相对集中的地区或网络中,在用户访问网站时,利用全局负载技术将用户的访问指向距离最近的工 ...

  3. 函数中的this与argument对象,以及argument中的callee与caller属性

    相关阅读:https://segmentfault.com/a/1190000015438195 相关阅读: https://zhuanlan.zhihu.com/p/23804247 相关阅读: h ...

  4. CENTOS手动安装修复python ,YUM CENTOS手动安装修复YUM

    CENTOS手动安装修复YUM  2019年3月8日  杨宇 Comments 0 Comment 目录 [hide] 一.问题场景 二.手动修复 2.1 下载 rpm 包 2.2 安装 rpm 包 ...

  5. Re.多项式除法/取模

    前言 emmm又是暂无 前置 多项式求逆 多项式除法/取模目的 还是跟之前一样顾名思义] 给定一个多项式F(x),请求出多项式Q(x)和R(x),满足F(x)=Q(x)∗G(x)+R(x),R项数小于 ...

  6. Python成绩

    # -*- coding: utf-8 -*- """ Spyder Editor This is a temporary script file. "&quo ...

  7. 公共DNS推荐及dns测速

    DNS在平时上网中扮演重要角色,如果不注意DNS的话,可能会导致网速慢.弹窗广告.网址打不开.打开不是自己想要的网站.劫持等一系列问题.针对DNS的问题,今天我们就来总结一下,看看哪个DNS服务器最好 ...

  8. 关于ajax的与后台Controller的交互 后台拿不到值

    话不多说 上代码 这是前段js的代码        传的两个参数    cLassid  和  userid $.ajax({ type:"post", url:"../ ...

  9. 爬虫框架Scrapy 之(二) --- scrapy文件

    框架简介 核心部分: 引擎.下载器.调度器 自定义部分: spider(自己建的爬虫文件).管道(pipelines.py) 目录结构 firstSpider firstSpider spiders ...

  10. Technocup 2019 - Elimination Round 2

    http://codeforces.com/contest/1031 (如果感觉一道题对于自己是有难度的,不要后退,懂0%的时候敲一遍,边敲边想,懂30%的时候敲一遍,边敲边想,懂60%的时候敲一遍, ...