动态代理
2019-10-25 07:10:20来源:博客园 阅读 ()
动态代理
第一种方式:Java提供的动态代理的方式
Java中有一个动态代理类对象的生成类Proxy,在这个类中有个方法:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)throws IllegalArgumentException
返回指定接口的代理类实例
将方法调用分派给指定的调用
处理程序。使用这个方法就可以生成一个动态代理对象
但是这个方法有三个参数,第三个参数就是定义一个自定义类型的处理器,处理将来代理类所代理的方法。
loader,拿到类的信息,方法....
interfaces,拿到接口
InvocationHandler实现类中有个方法是invoke,这个方法中的method就是由这两个参数反射得到的
proxy调用方法时就触发方法invoke
默认是给接口定义的所有方法加了功能,也就是说改造了所有接口实现方法
被代理的类是在程序运行的时候加载的,所以代理类是在运行的时候才构建的,invoke方法的参数在运行的时候才明确
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; //自定义类型的处理器,处理将来代理类所代理的方法 public class MyInvocationHandler implements InvocationHandler{ private MyLogger log = new MyLogger(); private Object target; public MyInvocationHandler(Object target) { this.target = target; } //如何来代理目标类的方法 /* * 参数1:将来生成的代理类对象; * 参数2:将来要代理的方法的镜像; * 参数3:将来所代理方法中需要的参数的镜像; * */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { String name = method.getName(); //添加额外功能 log.logger(name+"方法马上被执行!!!"); //调用目标类中的方法,用来真正实现逻辑功能 //target.method(args); Object invoke = method.invoke(target, args); //添加额外功能 return invoke; } }
InvocationHandler是由代理实例的调用处理程序实现的接口。
每个代理实例都有一个关联的调用处理程序。当在代理实例上调用方法时,方法调用被编码并发送到其调用处理程序的invoke方法。
Java动态代理的两个要素
实现统一接口,是在运行时动态实现的
产生关联,动态代理的关联是在自定义类型的处理器中处理的
JavaJDK实现动态代理的方式,就是构造一个实现代理类功能的对象
Cglib实现动态代理的方式,就是构造一个被代理类的子类,添加新的功能,继承核心功能
第二种方式:构造继承关系的Cglib方式
public class CglibProxy implements MethodInterceptor{ //创建出一个指定父类型的子类对象 public Object getProxy(Class c){ Enhancer enhancer = new Enhancer(); //设置谁是父类 enhancer.setSuperclass(c); //这里就是将当前类CglibProxy中的方法(intercept)及对象给了子类对象,也就是代理类对象 intercept雷同于Java中动态代理处理接口invocationHandler中的方法invoke方法作用差不多 enhancer.setCallback(this); //通过字节码技术动态创建子类实例 return enhancer.create(); } //intercept方法会拦截所有代理对象中方法的调用 //obj 参数:将来生成的代理对象 //method参数:将来代理对象所调用的方法的镜像 //args 参数:将来代理对象调用方法时所传的参数 //mproxy参数:该参数可以用来调用到父类中的方法 public Object intercept(Object obj, Method method, Object[] args,MethodProxy mproxy) throws Throwable { System.out.println("目标方法执行之前"); //调用父类中的方法 Object result = mproxy.invokeSuper(obj, args); System.out.println("目标方法执行之后"); return result; } }
原文链接:https://www.cnblogs.com/Magic-Li/p/11733135.html
如有疑问请与原作者联系
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- 代理项 Surrogate 2020-06-11
- Spring Boot 2.3.0 新特性Redis 拓扑动态感应 2020-06-11
- SpringBoot通过web页面动态控制定时任务的启动、停止、创建 2020-06-09
- Spring Cloud Gateway 扩展支持动态限流 2020-06-08
- CGLIB动态代理机制,各个方面都有写到 2020-06-04
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash