信息发布→ 登录 注册 退出

Java 动态代理的多种实现方式

发布时间:2026-01-11

点击量:
目录
  • 一、动态代理简介
  • 二、动态代理的多种实现
    • 1. 基于JDK的实现
    • 2. 基于cglib的实现
  • 三、为什么要有基于cglib的实现
    • 四、两种方式的适用场景
      • JDK动态代理
        • 优点
        • 缺点
        • 适用场景
      • cglib
        • 优点
        • 缺点
        • 适用场景

    一、动态代理简介

    优势:在不修改源码的情况下,对目标方法进行相应的增强。

    作用:完成程序功能之间的松耦合。

    二、动态代理的多种实现

    • JDK代理:基于接口的动态代理技术(缺点,目标对象必须有接口,如果没有接口,则无法完成动态代理的实现)
    • cglib代理:基于父类的动态代理技术

    两者的区别如图所示:

    1. 基于JDK的实现

    目标接口类:

    public interface TargetInterface {
    	public void save();
    	
    	public void print(String str);
    }
    

    目标类:

    public class Target implements TargetInterface{
    	
    	public void save() {
    		System.out.println("save running...");
    	}
    	
    	public void print(String str) {
    		System.out.println(str);
    	}
    
    }
    

    增强类:

    public class Advice {
    	public void before() {
    		System.out.println("前置增强");
    	}
    
    	public void after() {
    		System.out.println("后置增强");
    	}
    }
    

    测试类:

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    public class ProxyTest {
    
    	public static void main(String[] args) {
    		
    		//目标对象
    		final Target target = new Target();
    		
    		//增强对象
    		final Advice advice = new Advice();
    		
    		TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(
    				target.getClass().getClassLoader(), 					//目标对象类加载器
    				target.getClass().getInterfaces(), 						//目标对象相同的接口字节码对象数组
    				new InvocationHandler() {
    					//调用代理对象的任何方法,实质执行的都是invoke方法
    					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
    						advice.before();								//前置增强
    						Object invoke = method.invoke(target, args);	//执行目标方法
    						advice.after();									//后置增强
    						System.out.println();
    						return invoke;
    					}
    				});
    		
    		//代理对象的方法测试
    		proxyInstance.save();
    		
    		proxyInstance.print("JDK动态代理");
    	}
    
    }
    

    运行截图:

    2. 基于cglib的实现

    需要导入Jar包,如果是maven项目,则在pom.xml文件加入如下配置:

    <dependency>
    	<groupId>org.springframework</groupId>
    	<artifactId>spring-context</artifactId>
    	<version>4.2.4.RELEASE</version>
    </dependency>
    

    目标类:

    public class Target {
    	public void save() {
    		System.out.println("save running...");
    	}
    	
    	public void print(String str) {
    		System.out.println(str);
    	}
    }
    

    增强类:

    public class Advice {
    	
    	public void before() {
    		System.out.println("前置增强");
    	}
    
    	public void after() {
    		System.out.println("后置增强");
    	}
    
    }
    

    测试类:

    import java.lang.reflect.Method;
    
    import org.springframework.cglib.proxy.Enhancer;
    import org.springframework.cglib.proxy.MethodInterceptor;
    import org.springframework.cglib.proxy.MethodProxy;
    
    public class ProxyTest {
    
    	public static void main(String[] args) {
    		final Target target = new Target();
    		final Advice advice = new Advice();
    		
    		//返回值就是动态生成的代理对象,基于cglib
    		//创建增强器
    		Enhancer enhancer = new Enhancer();
    		
    		//设置父类(目标)
    		enhancer.setSuperclass(Target.class);
    		
    		//设置回调
    		enhancer.setCallback(new MethodInterceptor() {
    			public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{
    				advice.before();
    				Object invoke = method.invoke(target, obj);
    				advice.after();
    				System.out.println();
    				return invoke;
    			}
    		});
    		
    		//创建代理对象
    		Target proxy = (Target)enhancer.create();
    		
    		//测试代理方法
    		proxy.save();
    		proxy.print("基于cglib实现动态规划");
    		
    	}
    
    }
    

    运行截图:

    三、为什么要有基于cglib的实现

    使用JDK动态代理实现时,最大限制是被增强对象必须实现接口,并且增强的方法只能是接口中声明的方法。但在实际的项目中,可能总是存在对不实现业务接口的对象进行增强的需求,这时JDK动态代理将无能为力。

    四、两种方式的适用场景

    JDK动态代理

    优点

    • 不依赖第三方jar包, 使用方便
    • 随着JDK的升级,JDK动态代理的性能在稳步提升

    缺点

    • 只能代理实现了接口的类
    • 执行速度较慢

    适用场景

    • 如果你的程序需要频繁、反复地创建代理对象,则JDK动态代理在性能上更占优。

    cglib

    优点

    由于是动态生成字节码实现代理,因此代理对象的执行速度较快, 约为JDK动态代理的1.5 ~ 2倍
    可以代理没有实现接口的对象

    缺点

    • 不能代理final类
    • 动态生成字节码虽然执行较快,但是生成速度很慢,根据网上一些人的测试结果,cglib创建代理对象的速度要比JDK慢10 ~ 15倍。

    适用场景

    • 不需要频繁创建代理对象的应用,如Spring中默认的单例bean,只需要在容器启动时生成一次代理对象。

    以上就是Java 动态代理的多种实现方式的详细内容,更多关于Java 动态代理的实现的资料请关注其它相关文章!

    在线客服
    服务热线

    服务热线

    4008888355

    微信咨询
    二维码
    返回顶部
    ×二维码

    截屏,微信识别二维码

    打开微信

    微信号已复制,请打开微信添加咨询详情!