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

 更新时间:2021年06月07日 09:25:13   作者:卡修斯  
动态代理实际上是JVM在运行期动态创建class字节码并加载的过程。本文讲述了Java 动态代理的多种实现方式,感兴趣的朋友可以选择适合自己的方式

一、动态代理简介

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

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

二、动态代理的多种实现

  • 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 动态代理的实现的资料请关注脚本之家其它相关文章!

相关文章

  • springboot2启动时执行,初始化(或定时任务)servletContext问题

    springboot2启动时执行,初始化(或定时任务)servletContext问题

    这篇文章主要介绍了springboot2启动时执行,初始化(或定时任务)servletContext问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • SpringCloud负载均衡spring-cloud-starter-loadbalancer解读

    SpringCloud负载均衡spring-cloud-starter-loadbalancer解读

    这篇文章主要介绍了SpringCloud负载均衡spring-cloud-starter-loadbalancer使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-03-03
  • Mybatis-Plus根据时间段去查询数据的实现示例

    Mybatis-Plus根据时间段去查询数据的实现示例

    这篇文章主要介绍了Mybatis-Plus根据时间段去查询数据的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • Java JDBC连接数据库常见操作总结

    Java JDBC连接数据库常见操作总结

    这篇文章主要介绍了Java JDBC连接数据库常见操作,结合实例形式总结分析了java基于jdbc连接mysql、Oracle数据库及连接池相关操作技巧,需要的朋友可以参考下
    2019-03-03
  • Java深入讲解static操作符

    Java深入讲解static操作符

    static关键字基本概念我们可以一句话来概括:方便在没有创建对象的情况下来进行调用。也就是说:被static关键字修饰的不需要创建对象去调用,直接根据类名就可以去访问,让我们来了解一下你可能还不知道情况
    2022-07-07
  • Java实现将byte[]转换为File对象

    Java实现将byte[]转换为File对象

    这篇文章将通过一个简单的例子为大家演示Java如何实现 byte[] 转换为 File 对象,并将其上传到外部服务器,感兴趣的小伙伴可以跟随小编一起学习一下
    2025-03-03
  • SpringBoot MyBatis保姆级整合教程

    SpringBoot MyBatis保姆级整合教程

    因为Spring Boot框架开发的便利性,所以实现Spring Boot与数据访问层框架(例如MyBatis)的整合非常简单,主要是引入对应的依赖启动器,并进行数据库相关参数设置即可
    2022-06-06
  • Java LinkedList的实现原理图文详解

    Java LinkedList的实现原理图文详解

    今天小编就为大家分享一篇关于Java LinkedList的实现原理图文详解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • 深入理解java线程通信

    深入理解java线程通信

    开发中不免会遇到需要所有子线程执行完毕通知主线程处理某些逻辑的场景。或者是线程 A 在执行到某个条件通知线程 B 执行某个操作。下面我们来一起学习如何解决吧
    2019-05-05
  • MyBatis中动态sql的实现方法示例

    MyBatis中动态sql的实现方法示例

    这篇文章主要给大家介绍了关于MyBatis中动态sql的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-11-11

最新评论