java动态代理详解
java动态代理详解
摘要
本文动态代理得意义、主要介绍动态代理得实现原理以及由动态代理引申出来的一些知识点。
插曲
最近在研究javamelody实现的原理,发现他对JDBC的监控就是通过动态代理实现的。由于之前对于动态代理只是大概知道怎么回事,没有细致的去研究,所以上网百度了一下。发现网上的东西要么注重原理而忽略应用场景导致空泛、要么注重场景而忽略原理、要么就是只有基于接口的动态代理而没有基于cglib的。因此这里本文尽量做到大而全。其实想总结一下的原因是公司进行代码review的时候,老大提出同一个类中一个方法调用本类其他方法,其他方法的事务不会生效,本质上我是持怀疑态度的。当时我是出于基于Cglib代理的角度考虑,而实际不会生效是基于动态代理的方式,采用cglib还是会生效,后面会讲到。本人作文比较推崇简约易懂的方式,尽量避免过于斯文的名词出现。
一、动态代理的意义
首先明白一点,动态代理就是用来生成代理对象的。我们知道传统的代理模式,通常是先定义一个代理类,该代理类需要持有目标对象(也有叫被代理对象,我觉得都行吧)。假设我们有1000个不同的目标对象(这1000个对象不是同一个类),那么我们需要预先定义1000个代理类,这是我们不能容忍的。于是乎,动态代理就出现了,它本质上是生成一个外表上和目标对象一样的代理对象,然后当我们调用代理对象的方法的时候,实际上它在他的方法里面去调用了目标对象对应的同名方法。
二、动态代理设计的核心思想
其实不要把这些设计想得多么高尚,假如我是动态代理设计的作者,由动态代理的意义部分我们知道,我们就是要想尽一切办法,通过目标对象生成代理对象,然后让代理对象的方法调用作用到目标对象的方法调用。没错动态代理的核心思想就是这么简单。比如目标类为Person,Person有一个方法叫做purchase(),此方法用于购物。我们期望purchase()方法有代理类去做处理,比如在购物前记录下购买了哪些东西。我们知道在使用一个类之前,是需要创建一个对象的,我们就在创建的地方动手脚。所以你看到了JDK动态Proxy.newInstance()的方式,也领略过Spring的Enhancer.create()。个人比较喜欢cglib的优雅、干净、利落。吐槽一下JDK的InvocationHandler像极了恶心的中间商。下面是JDK动态代理UML示意图
三、JDK动态代理
1,原理
在了解动态代理之前,我们需要了解Java字节码。如果不熟悉Java字节码,你可以理解为通过代码动态生成一个.java文件,然后将其编译为class文件加载到内存中。接下来JDK中的动态代理要做的事情就是怎么去生成一个ProxyPerson字节码文件。其实它就是在生成字节码的时候,持有了InvocationHandler对象,然后去实现了ProxyPerson对应的接口。在该接口的所有实现方法中,只做了一件事情就是调用invocationHandler.invoke()方法。从代码层面来看如下所示:
复制代码
public class ProxyPerson implements Purchase{
static{
Method method;// 接口的方法
Object[] args;// 接口参数
}
InvocationHandler handler;
public ProxyPerson(InvocationHandler handler){
this.handler = handler;
}
@overrde
public purchage(){
this,handler.invoke(this,method,args);
}
}
复制代码
那么上面这段代码是在什么时候生成的呢?
Proxy.newProxyInstance()
在我们调用JDK上面的这个方法的时候,底层就会去生成一个ProxyPerson字节码。知道了原理我们来解答一下JDK动态代理为何只能基于接口代理而不能基于类呢?
1),受限于字节码的生成方式,JDK本身就是基于InvocationHandler去做的代理中转。我们看到代理对象的方法调用于目标对象的调用没有半毛球关系,调用目标对象是我们自己在invoke方法里面完成的。
2),受限于同名的方法只能被向上转型成功的对象调用。比如有两个类Boy与Girl,他们都实现了接口Purchase,如果我们先获取到Girl的purchase()方法method,我们通过method.invoke(new Boy())这样必定会报错。但是如果我们获取到Purchase接口purchase()方法method,我们通过method.invoke(new Boy())这样是ok的,因为new Boy()可以向上转型为Purchase。
2,应用
比如无论是传统的MVC模型还是DDD模型,都离不开Service。我们知道Service方法使用@Transactional是可以开启事务控制的。那么这种注解式事务是如何实现的呢? 其实在工程启动的时候,我们就会有一个Bean的后置处理器去检查所有Bean一旦发现Bean的方法上有事务注解,他就通过Proxy.newInstance()去创建一个代理对象,将代理对象进行返回注入,而抛弃原本应该注入到容器的对象。所以我们看起来通过容器拿到的Service其实已经是代理对象了。在调用目标对象前,开启编程式事务即可。
四、cglib动态代理
有了上面的知识,我们要有对于cglib而言只是在生成字节码上面动手脚的觉悟。下面直观感受与一下生成过程
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public static void main(final String[] args) {
Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(Boy.class); enhancer.setCallback(new MethodInterceptor(){ @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("proxy method "+ method.getName()); if(method.getAnnotation(Transactional.class)!=null){ System.out.println(method.getName()+"发现注解"); } return methodProxy.invokeSuper(o,args); } }); Boy proxy = (Boy) enhancer.create(); proxy.test();
}
public static class Boy{
public void run(){ System.out.println("run..."); } @Transactional public void walk(){ System.out.println("walk..."); } @Transactional public void test(){ System.out.println("test..."); walk(); }
}
可以看到cglib是基于继承的方式进行字节码动态生成。它在子类的实现中,只是调用了注入的methodIntercptor.interceptor()方法。具体字节码实现细节,这里不在深究。我们在这里探讨一下,为什么cglib可以使同一个service方法中的其他带有事务注解的事务生效?因为基于继承的动态代理,本质发起上调用的代理对象可以向上转型为原本的目标对象,所以它可以直接通过代理对象去调目标对象方法。
原文地址https://www.cnblogs.com/enjoyall/p/11324671.html

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
在阿里Java大牛们都是这样对Java项目代码分层的
作者:咖啡拿铁代码分层,对于任何一个Java开发来说应该都不陌生。一个好的层次划分不仅可以能使代码结构更加清楚,还可以使项目分工更加明确,可读性大大提升,更加有利于后期的维护和升级。从另外一个角度来看,好的代码分层架构,应该是可以很好的匹配上单一职责原则的。这样就可以降低层与层之间的依赖,还能最大程度的复用各层的逻辑。本文就来介绍下Java 项目的代码到底应该如何分层。 1.背景 说起应用分层,大部分人都会认为这个不是很简单嘛 就controller,service, mapper三层。看起来简单,很多人其实并没有把他们职责划分开,在很多代码中,controller做的逻辑比service还多,service往往当成透传了,这其实是很多人开发代码都没有注意到的地方,反正功能也能用,至于放哪无所谓呗。这样往往造成后面代码无法复用,层级关系混乱,对后续代码的维护非常麻烦。的确在这些人眼中分层只是一个形式,前辈们的代码这么写的,其他项目代码这么写的,那么我也这么跟着写。但是在真正的团队开发中每个人的习惯都不同,写出来的代码必然带着自己的标签,有的人习惯controller写大量的业务逻辑,有...
- 下一篇
物联网平台数据解析使用说明
1、注意事项(请务必关注) a、要使用数据解析功能,创建产品时,数据类型必须选择透传/自定义 b、只有通过特定Topic上下行的数据才会进行数据解析 上行Topic:/sys/${productKey}/${deviceName}/thing/model/up_raw 下行Topic:/sys/${productKey}/${deviceName}/thing/model/down_raw c、目前,只支持使用JavaScript语言编写解析脚本,且脚本中要定义两个方法:protocolToRawData和rawDataToProtocol protocolToRawData方法的入参为jsonObj对象,出参为byte[ ]数组 rawDataToProtocol方法的入参为byte[ ]数组,出参为jsonObj对象 2、数据解析使用示例 脚本编辑和测试过程可以参考阿里云官方文档https://help.aliyun.com/document_detail/114621.html 3、数据解析流程(以属性上报为例) step1:设备端发布自定义消息到topic中,topic为/sy...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS6,CentOS7官方镜像安装Oracle11G
- Docker安装Oracle12C,快速搭建Oracle学习环境
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- SpringBoot2全家桶,快速入门学习开发网站教程
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- MySQL8.0.19开启GTID主从同步CentOS8
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- CentOS8编译安装MySQL8.0.19
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程