Java 动态代理与CGLIB详细介绍
发布时间 - 2026-01-10 23:09:59 点击率:次静态代理模式

因为需要对一些函数进行二次处理,或是某些函数不让外界知道时,可以使用代理模式,通过访问第三方,间接访问原函数的方式,达到以上目的。
interface Hosee{
String sayhi();
}
class Hoseeimpl implements Hosee{
@Override
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class HoseeProxy implements Hosee{
Hosee h;
public HoseeProxy(Hosee h)
{
this.h = h;
}
@Override
public String sayhi()
{
System.out.println("I'm proxy!");
return h.sayhi();
}
}
public class StaticProxy
{
public static void main(String[] args)
{
Hoseeimpl h = new Hoseeimpl();
HoseeProxy hp = new HoseeProxy(h);
System.out.println(hp.sayhi());
}
}
1.1 静态代理的弊端
如果要想为多个类进行代理,则需要建立多个代理类,维护难度加大。
仔细想想,为什么静态代理会有这些问题,是因为代理在编译期就已经决定,如果代理哪个发生在运行期,这些问题解决起来就比较简单,所以动态代理的存在就很有必要了。
2.动态代理
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
interface HoseeDynamic
{
String sayhi();
}
class HoseeDynamicimpl implements HoseeDynamic
{
@Override
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class MyProxy implements InvocationHandler
{
Object obj;
public Object bind(Object obj)
{
this.obj = obj;
return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable
{
System.out.println("I'm proxy!");
Object res = method.invoke(obj, args);
return res;
}
}
public class DynamicProxy
{
public static void main(String[] args)
{
MyProxy myproxy = new MyProxy();
HoseeDynamicimpl dynamicimpl = new HoseeDynamicimpl();
HoseeDynamic proxy = (HoseeDynamic)myproxy.bind(dynamicimpl);
System.out.println(proxy.sayhi());
}
}
类比静态代理,可以发现,代理类不需要实现原接口了,而是实现InvocationHandler。通过
Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
来动态生成一个代理类,该类的类加载器与被代理类相同,实现的接口与被代理类相同。
通过上述方法生成的代理类相当于静态代理中的代理类。
这样就实现了在运行期才决定代理对象是怎么样的,解决了静态代理的弊端。
当动态生成的代理类调用方法时,会触发invoke方法,在invoke方法中可以对被代理类的方法进行增强。
通过动态代理可以很明显的看到它的好处,在使用静态代理时,如果不同接口的某些类想使用代理模式来实现相同的功能,将要实现多个代理类,但在动态代理中,只需要一个代理类就好了。
除了省去了编写代理类的工作量,动态代理实现了可以在原始类和接口还未知的时候,就确定代理类的代理行为,当代理类与原始类脱离直接联系后,就可以很灵活地重用于不同的应用场景中。
2.1 动态代理的弊端
代理类和委托类需要都实现同一个接口。也就是说只有实现了某个接口的类可以使用Java动态代理机制。但是,事实上使用中并不是遇到的所有类都会给你实现一个接口。因此,对于没有实现接口的类,就不能使用该机制。
而CGLIB则可以实现对类的动态代理
2.2 回调函数原理
上文说了,当动态生成的代理类调用方法时,会触发invoke方法。
很显然invoke方法并不是显示调用的,它是一个回调函数,那么回调函数是怎么被调用的呢?
上述动态代理的代码中,唯一不清晰的地方只有
Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
.getClass().getInterfaces(), this);
跟踪这个方法的源码,可以看到程序进行了验证、优化、缓存、同步、生成字节码、显示类加载等操作,前面的步骤并不是我们关注的重点,而最后它调用了
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
proxyName, interfaces);
该方法用来完成生成字节码的动作,这个方法可以在运行时产生一个描述代理类的字节码byte[]数组。
在main函数中加入
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
加入这句代码后再次运行程序,磁盘中将会产生一个名为”$Proxy().class”的代理类Class文件,反编译(反编译工具我使用的是 JD-GUI )后可以看见如下代码:
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 HoseeDynamic
{
private static Method m1;
private static Method m3;
private static Method m0;
private static Method m2;
public $Proxy0(InvocationHandler paramInvocationHandler)
throws
{
super(paramInvocationHandler);
}
public final boolean equals(Object paramObject)
throws
{
try
{
return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final String sayhi()
throws
{
try
{
return (String)this.h.invoke(this, m3, null);
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final int hashCode()
throws
{
try
{
return ((Integer)this.h.invoke(this, m0, null)).intValue();
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
public final String toString()
throws
{
try
{
return (String)this.h.invoke(this, m2, null);
}
catch (Error|RuntimeException localError)
{
throw localError;
}
catch (Throwable localThrowable)
{
throw new UndeclaredThrowableException(localThrowable);
}
}
static
{
try
{
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
m3 = Class.forName("HoseeDynamic").getMethod("sayhi", new Class[0]);
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
return;
}
catch (NoSuchMethodException localNoSuchMethodException)
{
throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
}
catch (ClassNotFoundException localClassNotFoundException)
{
throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
}
}
}
动态代理类不仅代理了显示定义的接口中的方法,而且还代理了java的根类Object中的继承而来的equals()、hashcode()、toString()这三个方法,并且仅此三个方法。
可以在上述代码中看到,无论调用哪个方法,都会调用到InvocationHandler的invoke方法,只是参数不同。
2.3 动态代理与静态代理的区别
Proxy类的代码被固定下来,不会因为业务的逐渐庞大而庞大;
可以实现AOP编程,这是静态代理无法实现的;
解耦,如果用在web业务下,可以实现数据层和业务层的分离。
动态代理的优势就是实现无侵入式的代码扩展。 静态代理这个模式本身有个大问题,如果类方法数量越来越多的时候,代理类的代码量是十分庞大的。所以引入动态代理来解决此类问题
3. CGLIB
cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
class CGlibHosee
{
public String sayhi()
{
return "Welcome oschina hosee's blog";
}
}
class CGlibHoseeProxy
{
Object obj;
public Object bind(final Object target)
{
this.obj = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(obj.getClass());
enhancer.setCallback(new MethodInterceptor()
{
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable
{
System.out.println("I'm proxy!");
Object res = method.invoke(target, args);
return res;
}
});
return enhancer.create();
}
}
public class CGlibProxy
{
public static void main(String[] args)
{
CGlibHosee cGlibHosee = new CGlibHosee();
CGlibHoseeProxy cGlibHoseeProxy = new CGlibHoseeProxy();
CGlibHosee proxy = (CGlibHosee) cGlibHoseeProxy.bind(cGlibHosee);
System.out.println(proxy.sayhi());
}
}
cglib需要指定父类和回调方法。当然cglib也可以与Java动态代理一样面向接口,因为本质是继承。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# Java
# 动态代理与CGLIB
# 动态代理
# Java基础之动态代理Cglib详解
# 详解Java Cglib动态代理
# Java使用JDK与Cglib动态代理技术统一管理日志记录
# JAVA中的静态代理、动态代理以及CGLIB动态代理总结
# Java CGLib动态代理机制(全面解析)
# java 中动态代理(JDK
# cglib)实例代码
# 深入理解java动态代理的两种实现方式(JDK/Cglib)
# 浅谈Java代理(jdk静态代理、动态代理和cglib动态代理)
# java动态代理和cglib动态代理示例分享
# java动态代理(jdk与cglib)详细解析
# Java cglib动态代理原理分析
# 多个
# 回调
# 的是
# 实现了
# 可以使用
# 可以实现
# 来实现
# 这是
# 反编译
# 加载
# 会有
# 是因为
# 给你
# 有个
# 将会
# 子类
# 不需要
# 说了
# 是怎么
# 而来
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
微信小程序 五星评分(包括半颗星评分)实例代码
Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】
Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明
香港服务器网站卡顿?如何解决网络延迟与负载问题?
JavaScript模板引擎Template.js使用详解
CSS3怎么给轮播图加过渡动画_transition加transform实现【技巧】
高端云建站费用究竟需要多少预算?
香港网站服务器数量如何影响SEO优化效果?
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
北京的网站制作公司有哪些,哪个视频网站最好?
Laravel怎么自定义错误页面_Laravel修改404和500页面模板
JavaScript数据类型有哪些_如何准确判断一个变量的类型
Python3.6正式版新特性预览
HTML 中如何正确使用模板变量为元素的 name 属性赋值
Laravel怎么进行数据库事务处理_Laravel DB Facade事务操作确保数据一致性
在Oracle关闭情况下如何修改spfile的参数
电商网站制作价格怎么算,网上拍卖流程以及规则?
Laravel Admin后台管理框架推荐_Laravel快速开发后台工具
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
音响网站制作视频教程,隆霸音响官方网站?
Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】
如何制作公司的网站链接,公司想做一个网站,一般需要花多少钱?
Laravel Fortify是什么,和Jetstream有什么关系
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
Laravel distinct去重查询_Laravel Eloquent去重方法
如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?
laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
如何注册花生壳免费域名并搭建个人网站?
Laravel如何创建自定义中间件?(Middleware代码示例)
Android自定义listview布局实现上拉加载下拉刷新功能
Edge浏览器如何截图和滚动截图_微软Edge网页捕获功能使用教程【技巧】
使用豆包 AI 辅助进行简单网页 HTML 结构设计
,在苏州找工作,上哪个网站比较好?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】
网站制作报价单模板图片,小松挖机官方网站报价?
Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率
Laravel如何从数据库删除数据_Laravel destroy和delete方法区别
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
如何在Windows服务器上快速搭建网站?
PHP正则匹配日期和时间(时间戳转换)的实例代码
Laravel如何创建自定义Artisan命令?(代码示例)
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
清除minerd进程的简单方法
如何在云虚拟主机上快速搭建个人网站?
微信小程序 HTTPS报错整理常见问题及解决方案
Laravel Session怎么存储_Laravel Session驱动配置详解
Linux网络带宽限制_tc配置实践解析【教程】

