Java动态代理全面分析
代理模式
解说:给某一个对象提供一个代理,并由代理对象控制对原对象的引用;
代理模式需要以下几个角色:
1 主题:规定代理类和真实对象共同对外暴露的接口;
2 代理类:专门代理真实对象的类;
3 真实对象:需要被代理的对象;
代理解决的主要的业务就是需要在 真实对象的某个接口 前后处理一些事情,框架中多会用到这种功能,比如 打日志、记录时间等
静态代理
静态代理是指自己动手编写代码实现代理类;
优点:业务类只需要关注业务逻辑本身,保证了业务类的重用性。这是代理的共有优点。
缺点:每一个真实对象都需要一个具体的代理类,不能做到可重用;
静态代理比较简单,下边用代码来具体说明;
主题接口:IAnimal
public interface IAnimal {
/**
* 动物叫
*/
void bark();
}
真实对象:Dog
public class Dog implements IAnimal { private String name; public Dog(String name) {
this.name = name;
} @Override
public void bark() {
System.out.println(this.name + " bark:wang wang wang ... ");
}
}
代理:DogProxy
public class DogProxy implements IAnimal {
private Dog dog; public DogProxy(Dog dog) {
this.dog = dog;
} @Override
public void bark() {
long l = System.currentTimeMillis();
System.out.println("dog will bark...");
this.dog.bark();
System.out.println("dog has barked which takes " + (System.currentTimeMillis() - l) + " ms !");
}
}
静态代理使用:
public class StaticProxyTest { public static void main(String[] args) { IAnimal dog = new Dog("大黄"); IAnimal dogProxy = new DogProxy(dog); dogProxy.bark();
}
}
代理和真实对象对外暴露一致
动态代理
动态代理是指在运行时动态生成代理类;
jdk
要使用Java中原生的动态代理,需要用到以下几个类和接口
- 接口InvocationHandler
- Proxy类
我们还是用静态代理用到的代码:主题接口IAnimal和真实对象Dog不变,去掉DogProxy和StaticProxyTest,增加以下代码
DogProxyInvocationHandler
public class DogProxyInvocationHandler implements InvocationHandler {
private Object animal; public DogProxyInvocationHandler(Object animal) {
this.animal = animal;
} @Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("proxy class:" + proxy.getClass() + ",class:" + getClass() + ",method:" + method);
Object obj = method.invoke(animal, args);
System.out.println("obj:" + obj);
return obj;
}
}
DynamicProxyTest
public class DynamicProxyTest { public static void main(String[] args) { IAnimal dog = new Dog("大黄"); InvocationHandler invocationHandler = new DogProxyInvocationHandler(dog);
IAnimal animal = (IAnimal) Proxy.newProxyInstance(dog.getClass().getClassLoader(),
dog.getClass().getInterfaces(),
invocationHandler);
animal.bark();
} }
可以看出:Java动态代理 我们必须有真实对象,实现了InvocationHandler接口的自己的处理类,然后通过Proxy生成代理类
输出如下:
proxy class:class com.sun.proxy.$Proxy0,class:class com.shock.base.proxy.dynamic.DogProxyInvocationHandler,method:public abstract void com.shock.base.proxy.dynamic.IAnimal.bark()
obj:null
这里动态代理的优势相比静态代理为:即使真实对象有N个接口,我们的invocationHandler只需要一个Invoke方法即可!
这里有几个问题:
1 动态代理生成的class name为什么是 $Proxy0 ?
如图,相关变量如下:
以上便可以解决相关问题
2 动态代理生成的代理类到底是什么样子的?生成代理类的关键接口是什么?为何调用真实对象的某个接口会进入invoke方法?
通过分析源码:我们知道 Proxy.newProxyInstance → Proxy.getProxyClass0 → WeakCache.get → WeakCache.Factory.get → Proxy.ProxyClassFactory.apply → ProxyGenerator.generateProxyClass
最终生成了一个 byte[] 类型的 class类;这样byte[] 比较抽象 ,我们想看到该怎么办?可以通过下边的代码生成Proxy0
int accessFlags = Modifier.PUBLIC | Modifier.FINAL; byte[] bytes = ProxyGenerator.generateProxyClass("com.sun.proxy.$Proxy0",
new Class[]{IAnimal.class},
accessFlags);
FileOutputStream fileOutputStream = new FileOutputStream(new File(
"~/work/$Proxy0.class")); fileOutputStream.write(bytes);
fileOutputStream.flush();
fileOutputStream.close();
生成的代码如下:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
// package com.sun.proxy; import com.shock.base.proxy.dynamic.IAnimal;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException; public final class $Proxy0 extends Proxy implements IAnimal {
private static Method m1;
private static Method m2;
private static Method m3;
private static Method m0; public $Proxy0(InvocationHandler var1) throws {
super(var1);
} public final boolean equals(Object var1) throws {
try {
return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue();
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
} 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);
}
} public final void bark() throws {
try {
super.h.invoke(this, m3, (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)).intValue();
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
} static {
try {
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")});
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
m3 = Class.forName("com.shock.base.proxy.dynamic.IAnimal").getMethod("bark", new Class[0]);
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}
以上代码为我们解答了红色的问题。虽然代码是这样的 代理类集成了 Proxy类,但是如果想要验证 如何验证呢?
cglib
cglib是什么?CGLIB is a powerful, high performance code generation library.
特点简单说:
- CGLib (Code Generation Library) 是一个强大的,高性能,高质量的Code生成类库;
- 它可以在运行期扩展Java类与实现Java接口;
- CGLib 比 Java 的 java.lang.reflect.Proxy 类更强的在于它不仅可以接管接口类的方法,还可以接管普通类的方法。
- CGLib 的底层是Java字节码操作框架 —— ASM
引入JAR包支持,如下:
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.2.4</version>
</dependency>
目前最新的版本是 3.2.4,CGLib的package分布和作用如下:
- net.sf.cglib.core:底层字节码处理类,他们大部分与ASM有关系,对其进行封装,更易于使用;
- net.sf.cglib.transform:编译期或运行期类和类文件的转换;
- net.sf.cglib.proxy:实现创建代理和方法拦截器的类;
- net.sf.cglib.reflect:实现快速反射的类;
- net.sf.cglib.util:集合排序工具类;
- net.sf.cglib.beans:JavaBean相关的工具类;
我们沿用上边的例子,来做下演示:
加入你有一个类Animal,打算对里边的所有方法进行包装,由于这个类没有实现接口,所以你无法使用jdk 动态代理
public class Animal { public void bark() {
System.out.println("i am 大黄!");
} public String singSong(String name) {
return name + " is singing!";
} public String testt(String name) {
return name + " is testing!";
}
}
你现在想要在testt方法输出前后不加任何内容,但是另外两个方法输出前后要加一个字符串,效果如下:
BEFORE
i am 大黄!
AFTER
===============================
BEFORE
老刘 is singing!
AFTER
===============================
大黄 is testing!
其中=======是分隔线,上边两个方法前后都改变了,但是最后一个方法则没做任何改变还是原生的。如何做到?
首先我们要定义一个拦截器,该拦截器实现了 cgLib的MethodInterceptor,如下:
public class AnimalWrapper implements MethodInterceptor { @Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("BEFORE");
Object obj2 = proxy.invokeSuper(obj, args);
System.out.println("AFTER");
return obj2;
}
}
再定义一个拦截器过滤器,如下:
public class ApiFilter implements CallbackFilter { @Override
public int accept(Method method) {
String name = method.getName();
if (name.length() == 5) {
return 1;
}
return 0;
}
}
接下来看下测试类:
public class CglibTest { public static void main(String[] args) { Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(Animal.class);
enhancer.setCallbacks(new Callback[]{new AnimalWrapper(), NoOp.INSTANCE});
enhancer.setCallbackFilter(new ApiFilter());
Animal animal = (Animal) enhancer.create(); animal.bark();
System.out.println("===============================");
animal.singSong("老刘");
System.out.println("===============================");
animal.testt("大黄");
}
}
CGLib通过 Enhancer、Callback、CallbackFilter就可以实现上述功能了。
深入代码:Enhancer → KeyFactory.Generator → AbstractClassGenerator → DefaultGeneratorStrategy.generate → KeyFactory.Generator.generateClass → ClassWriter.toByteArray 生成 class bytecode .
生成的代码反编译如下:
public class Animal$$EnhancerByCGLIB$$223151cf extends com.shock.base.proxy.cglib.Animal implements net.sf.cglib.proxy.Factory {
private boolean CGLIB$BOUND;
public static java.lang.Object CGLIB$FACTORY_DATA;
private static final java.lang.ThreadLocal CGLIB$THREAD_CALLBACKS;
private static final net.sf.cglib.proxy.Callback[] CGLIB$STATIC_CALLBACKS;
private net.sf.cglib.proxy.MethodInterceptor CGLIB$CALLBACK_0;
private net.sf.cglib.proxy.NoOp CGLIB$CALLBACK_1;
private static java.lang.Object CGLIB$CALLBACK_FILTER;
private static final java.lang.reflect.Method CGLIB$bark$0$Method;
private static final net.sf.cglib.proxy.MethodProxy CGLIB$bark$0$Proxy;
private static final java.lang.Object[] CGLIB$emptyArgs;
private static final java.lang.reflect.Method CGLIB$singSong$1$Method;
private static final net.sf.cglib.proxy.MethodProxy CGLIB$singSong$1$Proxy;
private static final java.lang.reflect.Method CGLIB$equals$3$Method;
private static final net.sf.cglib.proxy.MethodProxy CGLIB$equals$3$Proxy;
private static final java.lang.reflect.Method CGLIB$toString$4$Method;
private static final net.sf.cglib.proxy.MethodProxy CGLIB$toString$4$Proxy;
private static final java.lang.reflect.Method CGLIB$hashCode$5$Method;
private static final net.sf.cglib.proxy.MethodProxy CGLIB$hashCode$5$Proxy; static void CGLIB$STATICHOOK1() { /* compiled code */ } final void CGLIB$bark$0() { /* compiled code */ } public final void bark() { /* compiled code */ } final void CGLIB$singSong$1(java.lang.String s) { /* compiled code */ } public final void singSong(java.lang.String s) { /* compiled code */ } final boolean CGLIB$equals$3(java.lang.Object o) { /* compiled code */ } public final boolean equals(java.lang.Object o) { /* compiled code */ } final java.lang.String CGLIB$toString$4() { /* compiled code */ } public final java.lang.String toString() { /* compiled code */ } final int CGLIB$hashCode$5() { /* compiled code */ } public final int hashCode() { /* compiled code */ } public static net.sf.cglib.proxy.MethodProxy CGLIB$findMethodProxy(net.sf.cglib.core.Signature signature) { /* compiled code */ } public Animal$$EnhancerByCGLIB$$223151cf() { /* compiled code */ } public static void CGLIB$SET_THREAD_CALLBACKS(net.sf.cglib.proxy.Callback[] callbacks) { /* compiled code */ } public static void CGLIB$SET_STATIC_CALLBACKS(net.sf.cglib.proxy.Callback[] callbacks) { /* compiled code */ } private static final void CGLIB$BIND_CALLBACKS(java.lang.Object o) { /* compiled code */ } public java.lang.Object newInstance(net.sf.cglib.proxy.Callback[] callbacks) { /* compiled code */ } public java.lang.Object newInstance(net.sf.cglib.proxy.Callback callback) { /* compiled code */ } public java.lang.Object newInstance(java.lang.Class[] classes, java.lang.Object[] objects, net.sf.cglib.proxy.Callback[] callbacks) { /* compiled code */ } public net.sf.cglib.proxy.Callback getCallback(int i) { /* compiled code */ } public void setCallback(int i, net.sf.cglib.proxy.Callback callback) { /* compiled code */ } public net.sf.cglib.proxy.Callback[] getCallbacks() { /* compiled code */ } public void setCallbacks(net.sf.cglib.proxy.Callback[] callbacks) { /* compiled code */ }
}
该段代码最后实在没办法生成,是将CGLib代码源码下载,然后插入片段代码生成的。
Java动态代理全面分析的更多相关文章
- Java 动态代理机制分析及扩展
Java 动态代理机制分析及扩展,第 1 部分 王 忠平, 软件工程师, IBM 何 平, 软件工程师, IBM 简介: 本文通过分析 Java 动态代理的机制和特点,解读动态代理类的源代码,并且模拟 ...
- Java 动态代理机制分析及扩展,第 1 部分
Java 动态代理机制分析及扩展,第 1 部分 http://www.ibm.com/developerworks/cn/java/j-lo-proxy1/ 本文通过分析 Java 动态代理的机制和特 ...
- [转]Java 动态代理机制分析及扩展
引言 Java 动态代理机制的出现,使得 Java 开发人员不用手工编写代理类,只要简单地指定一组接口及委托类对象,便能动态地获得代理类.代理类会负责将所有的方法调用分派到委托对象上反射执行,在分派执 ...
- Java 动态代理机制分析及扩展--转
http://www.ibm.com/developerworks/cn/java/j-lo-proxy1/#icomments http://www.ibm.com/developerworks/c ...
- 设计模式学习——JAVA动态代理原理分析
一.JDK动态代理执行过程 上一篇我们讲了JDK动态代理的简单使用,今天我们就来研究一下它的原理. 首先我们回忆下上一篇的代码: public class Main { public static v ...
- 彻底理解JAVA动态代理
代理设计模式 定义:为其他对象提供一种代理以控制对这个对象的访问. 代理模式的结构如下图所示. 动态代理使用 java动态代理机制以巧妙的方式实现了代理模式的设计理念. 代理模式示例代码 public ...
- Java动态代理原理及其简单应用
概念 代理对象和被代理对象一般实现相同的接口,调用者与代理对象进行交互.代理的存在对于调用者来说是透明的,调用者看到的只是接口.代理对象则可以封装一些内部的处理逻辑,如访问控制.远程通信.日志.缓存等 ...
- 一文读懂Java动态代理
作者 :潘潘 日期 :2020-11-22 事实上,对于很多Java编程人员来说,可能只需要达到从入门到上手的编程水准,就能很好的完成大部分研发工作.除非自己强主动获取,或者工作倒逼你学习,否则我们好 ...
- java动态代理实现与原理详细分析
关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式 代理模式是常用的java设计模式, ...
随机推荐
- C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图
C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图 +BIT祝威+悄悄在此留下版了个权的信息说: 最近需要用OpenGL绘制文字,这是个很费时费力的事.一般的思路就是 ...
- 30个你必须记住的CSS选择符
所以你学会了基础的id,类和后代选择符,然后你就一直用它们了吗?如果是这样,你丢失了(css的)巨大的灵活性.在本文中提到的很多选择器属于CSS3规范的一部分,因此,只有在现代浏览器中才可使用. 1. ...
- C指针(二)
原文链接:http://www.orlion.ga/924/ 一.指针与const限定符 const限定符与指针结合起来常见的情况有一下几种: const int *a; int const *a; ...
- 深入理解javascript对象系列第二篇——属性操作
× 目录 [1]查询 [2]设置 [3]删除[4]继承 前面的话 对于对象来说,属性操作是绕不开的话题.类似于“增删改查”的基本操作,属性操作分为属性查询.属性设置.属性删除,还包括属性继承.本文是对 ...
- 【原创】开源Math.NET基础数学类库使用(16)C#计算矩阵秩
本博客所有文章分类的总目录:[总目录]本博客博文总目录-实时更新 开源Math.NET基础数学类库使用总目录:[目录]开源Math.NET基础数学类库使用总目录 上个月 ...
- Create Volume 操作(Part III) - 每天5分钟玩转 OpenStack(52)
本节是创建 Volume 的第三部分,也是最后一部分:cinder-volume 的处理过程. 第一部分和第二部分可以参考前面两个小节.cinder-volume 通过 driver 创建 volum ...
- 为什么要学习Linux
作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 本篇修改自我在知乎上回答的问题,为什么要学习 Linux? 我最开始用Linux是 ...
- MongoDB学习系列(2)--使用PHP访问MongoDB
第一部分:介绍 在Windows上安装最新MongoDB步骤非常的简单,这里不做介绍.但是如果你安装的时候没有将MongoDB作为服务运行,每次你都要使用cmd切换到指定的目录下,然后在cmd中启动M ...
- 窥探Swift之数组安全索引与数组切片
今天是元宵节,祝大家元宵节快乐!在Swift中的数组和字典中下标是非常常见的,数组可以通过索引下标进行元素的查询,字典可以通过键下标来获取相应的值.在使用数组时,一个常见的致命错误就是数组越界.如果在 ...
- 结婚虽易,终老不易:EntityFramework和AutoMapper的婚后生活
写在前面 我到底是什么? 越界的可怕 做好自己 后记 上一篇<恋爱虽易,相处不易:当EntityFramework爱上AutoMapper>文章的最后提到,虽然AutoMapper为了En ...