JDK动态代理源码

一、public static Object newProxyInstance ——> 调用下面这个方法
二、Class<?> cl = getProxyClass0(loader, intfs); ——> 这个方法从下面这个缓存对象中返回代理类Class对象
三、return proxyClassCache.get(loader, interfaces); ——> 这个缓存对象是一个成员变量
四、proxyClassCache = new WeakCache<>(new KeyFactory(), new ProxyClassFactory()); ——> 打开ProxyClassFactory这个类
五、private static final class ProxyClassFactory ——> 这是Proxy的一个内部类
六、public Class<?> apply(ClassLoader loader, Class<?>[] interfaces) { ——> 该内部类有一个apply方法,查看apply方法
1、确定代理类的名称
(2)确定代理类包名:得到传入接口的全限定名,截取接口包名作为代理类的包名;如果接口没有包名,把包名设置为com.sun.proxy
(3)确定代理类类名:类名前缀为§Proxy;后面加一个递增的数字,从0开始,如第一个代理类的名称为§Proxy0
(4)把代理类的包名和类名拼接起来
2、生成代理类字节码文件
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces); ——> 打开generateProxyClass()方法,查看generateClassFile()方法
七、generateClassFile() ——>
1、生成hashCode、equals、toString方法
2、生成构造方法,this.methods.add(this.generateConstructor());
3、生成接口里的方法private ProxyGenerator.MethodInfo generateMethod() throws IOException {


JDK动态代理分析

JDK动态代理在运行期创建接口的代理类并返回代理对象。

InvocationHandler可以看成是一个编织器。
Invocationhandler的invoke(Object proxy, Method method, Object[] args)方法,第一个参数是代理对象,一般不会用到;第二个参数method是JDK通过调用——我们调用Proxy的newProxyInstance方法时传入的第二个接口类型参数——的getMethods得来的;通过反射调用method的invoke方法来调用目标类的代码,将横切逻辑和业务逻辑编织在一起。
InvocationHandler还有一个成员变量,即目标对象,声明为Object类型即可,调用method方法时的invoke方法时要把这个Object传入进去。因为在调用Proxy的静态方法时已经传入了接口,所以JDK会把Object类型的成员变量向下转型为接口类型。


JDK动态代理实现

直接运行下面的代码,在D盘就可以看到生成的源码类$GameProxy.class

import sun.misc.ProxyGenerator;
import java.io.File;
import java.io.FileOutputStream;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy; public class ViewProxyClass {
public static void main(String[] args) throws Exception {
Moveable huiXiong = new Tank("灰熊坦克");
ClassLoader classLoader = huiXiong.getClass().getClassLoader();
Class<?>[] interfaces = huiXiong.getClass().getInterfaces();
Moveable proxy = (Moveable) Proxy.newProxyInstance(classLoader, interfaces, new TimeInvocationHandler(huiXiong));
System.out.println(proxy.getClass());
proxy.move();
proxy.stop(); byte[] bts = ProxyGenerator.generateProxyClass("$GameProxy", interfaces);
FileOutputStream fos = new FileOutputStream(new File("D:/$GameProxy.class"));
fos.write(bts);
fos.flush();
fos.close();
}
} interface Moveable {
public void move();
public void stop();
public String getName();
} class Tank implements Moveable {
private String name;
public Tank(String name) {
this.name = name;
}
@Override
public String getName() {
return name;
}
@Override
public void stop() {
System.out.println(this.name + " stop ...");
}
@Override
public void move() {
System.out.println(this.name + " run ...");
}
} class TimeInvocationHandler implements InvocationHandler {
private Object target;
public TimeInvocationHandler(Moveable target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("begin");
method.invoke(target, args);
System.out.println("end");
return proxy;
}
}

生成的代理类

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
// import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException; public final class $GameProxy extends Proxy implements Moveable {
private static Method m3;
private static Method m1;
private static Method m4;
private static Method m5;
private static Method m0;
private static Method m2; public $GameProxy(InvocationHandler var1) throws {
// // 代理类调用父类的构造方法,父类Proxy把InvocationHandler赋值给一个成员变量
super(var1);
} public final String getName() throws {
try {
return (String)super.h.invoke(this, m3, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} public final boolean equals(Object var1) throws {
try {
return (Boolean)super.h.invoke(this, m1, new Object[]{var1});
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
} public final void stop() throws {
try {
super.h.invoke(this, m4, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} public final void move() throws {
try {
super.h.invoke(this, m5, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} public final int hashCode() throws {
try {
return (Integer)super.h.invoke(this, m0, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} public final String toString() throws {
try {
return (String)super.h.invoke(this, m2, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} static {
try {
m3 = Class.forName("Moveable").getMethod("getName");
m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
m4 = Class.forName("Moveable").getMethod("stop");
m5 = Class.forName("Moveable").getMethod("move");
m0 = Class.forName("java.lang.Object").getMethod("hashCode");
m2 = Class.forName("java.lang.Object").getMethod("toString");
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}

CGLib动态代理

CGLib不需要目标类实现接口,而是在运行期首先获得目标类的字节码文件,然后采用字节码技术为目标类成生一个子类,并返回父类引用的子类对象,利用多态织入横切逻辑。

CGLib不能对目标类的private、final修饰的方法进行代理,因为父类的final方法不允许子类重写,父类的private方法不允许子类访问。


CGLib动态代理实现

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; public class ViewProxyClass {
public static void main(String[] args) throws Exception {
MyCGlib myCGlib = new MyCGlib();
Tank tank = (Tank)myCGlib.getProxy(Tank.class);
tank.move();
tank.stop();
}
} class Tank {
public Tank() {};
public void stop() {
System.out.println(" stop ...");
}
public void move() {
System.out.println(" run ...");
}
} class MyCGlib implements MethodInterceptor {
Enhancer enhancer = new Enhancer();
// 参数clazz是目标类的Class对象,此方法返回代理对象
public Object getProxy(Class clazz) throws Exception {
enhancer.setSuperclass(clazz);
enhancer.setCallback(this);
// 通过字节码规范动态创建代理类对象
return enhancer.create();
}
@Override
// 参数o是目标类实例,method是目标类方法的反射对象,objects是方法的入参,methodProxy是代理类方法
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("begin");
// 调用目标类也就是父类的方法
Object proxy = methodProxy.invokeSuper(o, objects);
System.out.println("end");
return proxy;
}
}
    <dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.1</version>
</dependency>

pom.xml


动态代理

JDK和CGLib都是动态代理。

在程序运行时,当代码执行到Proxy.newProxyInstance()时,使用反射机制——调用传入的接口Class对象的getMethods(),在内存中动态生成代理类。

在程序运行时,当代码执行到xxxxx.getProxy(xxx.class)时,使用字节码技术——先获得目标类的字节码文件,再根据这个字节码文件为其生成子类。

CGLib创建的代理对象的性能比GDK创建的代理对象的性能高10倍,GDK生成的代理对象的方法调用也要用到反射,比如invoke方法的第二个参数就是Method对象,生成Method,调用Method的方法都是反射调用,效率低。

但CGLib创建代理对象花费的时间比JDK要多8倍,使用字节码技术生成代理类不如直接调用反射的API生成代理类效率高。

单例代理或者具有实例池的代理适合于CGLib,反之适于GDK。

												

JDK动态代理、CGLib动态代理的更多相关文章

  1. Atitit 代理CGLIB 动态代理 AspectJ静态代理区别

    Atitit 代理CGLIB 动态代理 AspectJ静态代理区别 1.1. AOP 代理主要分为静态代理和动态代理两大类,静态代理以 AspectJ 为代表:而动态代理则以 spring AOP 为 ...

  2. 【Java入门提高篇】Day12 Java代理——Cglib动态代理

    今天来介绍另一种更为强大的代理——Cglib动态代理. 什么是Cglib动态代理? 我们先回顾一下上一篇的jdk动态代理,jdk动态代理是通过接口来在运行时动态创建委托类的代理对象,但是跟静态代理一样 ...

  3. Java的三种代理模式:静态代理/JDK动态代理/Cglib动态代理

    1.静态代理:需要定义接口或者父类,目标对象与代理对象均实现同一接口或继承同一父类. 2.JDK动态代理:需要目标对象实现一个接口,通过动态反射的机制,生成代理对象,实现同一个接口 3.Cglib动态 ...

  4. java 静态代理 JDK动态代理 Cglib动态代理

    下面以一个简单的银行账户为例讲述讲述动态代理. 设计一个银行账户类,包含用户的账户余额,实现查询和更新余额功能 这个系统用了一段时间,有客户要求对账说账户余额给弄错了?因为上面没有存取款记录,最后银行 ...

  5. Java之代理(jdk静态代理,jdk动态代理,cglib动态代理,aop,aspectj)

    一.概念 代理是什么呢?举个例子,一个公司是卖摄像头的,但公司不直接跟用户打交道,而是通过代理商跟用户打交道.如果:公司接口中有一个卖产品的方法,那么公司需要实现这个方法,而代理商也必须实现这个方法. ...

  6. JDK和CGLIB动态代理原理区别

    JDK和CGLIB动态代理原理区别 https://blog.csdn.net/yhl_jxy/article/details/80635012 2018年06月09日 18:34:17 阅读数:65 ...

  7. 静态代理,动态代理和CGLIB代理模式

    代理模式 一.概述 代理是一种模式,提供了对目标对象的间接访问方式,即通过代理访问目标对象.如此便于在目标实现的基础上增加额外的功能操作,前拦截,后拦截等,以满足自身的业务需求,同时代理模式便于扩展目 ...

  8. Java深入浅出系列(四)——深入剖析动态代理--从静态代理到动态代理的演化

    静态代理 如上图,在程序执行之前.程序猿就要编写Proxy.然后进行编译,即在程序执行之前,代理类的字节码文件就已经生成了(Proxy类的class文件已经存在了). 静态代理尽管在增强现有的接口业务 ...

  9. java 代理模式-静态代理与动态代理

    最近在研究SpringAOP,当然要学习AOP就要知道这么健硕.强大的功能的背后究竟隐藏着怎样不可告人的“秘密”?? 接下来就是查阅了许多资料详细的研究了一下Java的代理模式,感觉还是非常非常重要的 ...

  10. Java静态代理与动态代理 理解与应用场景

    角色 抽象角色:接口类 实现角色: 实现类 代理角色:代理实现的类,最终使用的对象 静态代理 1. 接口 /** * description * * @author 70KG * @date 2018 ...

随机推荐

  1. C#:WebBrowser控件的使用教程及相关问题整理

    推荐阅读: C#WebBrowser控件使用教程与技巧收集--苏飞收集 C# WebBrowser强制在本窗口打开,禁止在新窗口打开 C# WebBrowser禁止在新窗口打开,强制在本窗口打开(多种 ...

  2. MySQL查询表结构命令

    参考网址:https://www.cnblogs.com/zhangyuhang3/p/6873895.html 一.简单描述表结构,字段类型 desc tabl_name; desc tabl_na ...

  3. Install vsftpd on centos

    安装vsftpd程序. sudo yum -y install vsftpd 启动ftp服务. sudo service vsftp start 添加ftp用户,并设置密码. sudo useradd ...

  4. SQL SERVICE 拆分字符串的表值函数

    SQL代码 ALTER FUNCTION [dbo].[SplitToTable]( @SplitString nvarchar(max), @Separator nvarchar(10)=' ')R ...

  5. Ubuntu 16.04 服务器上配置使用 Docker

    Docker基础概念 在使用Docker之前,我们先了解下几个Docker的核心概念 Docker Daemon Docker引擎,就是运行在后台的一个守护进程,在我们启动它之后,我们就可以通过Doc ...

  6. Django使用Signals监测model字段变化发送通知

    上一篇文章<运维效率之数据迁移自动化>中讲到了工单通知,本文将介绍工单通知实现过程中的一些小技巧.所有演示均基于Django2.0 阅读此篇文章你可以: 解锁一个python if的使用新 ...

  7. epoll 触发模式

    Edge Triggered (ET):边缘触发只有数据到来,才触发,不管缓存区中是否还有数据.Level Triggered (LT):水平触发只要有数据都会触发. LT(level trigger ...

  8. 修改linux的ssh默认端口号22的方法

    一.修改配置文件 vi /etc/ssh/sshd_config 找到#Port 22 修改为自己要使用的端口号:Port 26000 然后 :x  退出保存 二.重启ssh服务 /etc/init. ...

  9. C字符串

    C字符串 C中的字符串是以空字符('\0')结尾的一个char数组,基本的实现字符串的方法有:字符串常量,字符串数组,char数组,char指针.字符串使用广泛,如与用户交互等处理自然语言的情况.C为 ...

  10. C# 在webapi项目中配置Swagger(最新版2017)

    这篇文章已经过时了: http://www.cnblogs.com/alunchen/p/6888002.html 不用那么繁琐的配置了,直接导入一个包就行了: Install-Package Swa ...