带你重新认识Java动态代理

 更新时间:2021年11月22日 15:45:46   作者:浪里小白龙nbw  
这篇文章主要为大家介绍了Java的动态代理,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

什么是动态代理?

动态代理就是,在程序运行期,创建目标对象的代理对象,并对目标对象中的方法进行功能性增强的一种技术。在生成代理对象的过程中,目标对象不变,代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间,对象中方法的动态拦截,在拦截方法的前后执行功能操作(也可以对原方法的参数进行操作)。

代理类在程序运行期间,创建的代理对象称之为动态代理对象。这种情况下,创建的代理对象,并不是事先在Java代码中定义好的。而是在运行期间,根据我们在动态代理对象中的“指示”,动态生成的。也就是说,你想获取哪个对象的代理,动态代理就会为你动态的生成这个对象的代理对象。动态代理可以对被代理对象的方法进行功能增强。有了动态代理的技术,那么就可以在不修改方法源码的情况下,增强被代理对象的方法的功能,在方法执行前后做任何你想做的事情。

特点:字节码随用随创建,随用随加载

作用:不修改源码的基础上对方法增强

正常类创建对象的过程:

在这里插入图片描述

动态代理创建代理对象的过程:

在这里插入图片描述

动态代理的常用两种方式:

1.基于接口的动态代理

提供者:JDK

使用JDK官方的Proxy类创建代理对象

注意:代理的目标对象必须实现接口(至少一个)

2.基于类的动态代理

提供者:第三方 CGLib

使用CGLib的Enhancer类创建代理对象

注意:被代理类不能用 final 修饰的类(最终类)。如果报 asmxxxx 异常,需要导入 asm.jar包

//JDK动态代理(基于接口的动态代理)
Proxy.newProxyInstance(三个参数);
	ClassLoader:类加载器
		它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。(固定写法)
	Class[]:字节码数组
		它是用于让代理对象和被代理对象有相同方法。(固定写法)
	InvocationHandler:用于提供增强的代码
		它是让我们写如何代理。我们一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须
		InvocationHandler该接口的实现类是谁用谁写,此时我们用就需要我们自己写

此处以一个演员的例子为例:

在很久以前,演员和剧组都是直接见面联系的。没有中间人环节。

而随着时间的推移,产生了一个新兴职业:经纪人(中间人),这个时候剧组再想找演员就需要通过经纪人来找了。下面我们就用代码演示出来。

package com.haust.service;
public interface IActor {
	/**
	* 基本演出
	* @param money
	*/
	public void basicAct(float money);
	/**
	* 危险演出
	* @param money
	*/
	public void dangerAct(float money);
}
package com.haust.serviceImpl;
import com.haust.service.IActor;
public class Actor implements IActor {
	/**
	* 一个演员
	*/
	//实现了接口,就表示具有接口中的方法实现。即:符合经纪公司的要求
	@Override
	public void basicAct(float money) {
		System.out.println("拿到钱,开始基本的表演:"+money);
	}
	@Override
	public void dangerAct(float money) {
		System.out.println("拿到钱,开始危险的表演:"+money);
	}
}
package com.haust.test;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import com.haust.service.IActor;
import com.haust.serviceImpl.Actor;
public class Client {
	public static void main(String[] args) {
		//一个剧组找演员:
		final Actor actor = new Actor();//被代理的类
		/**
		* 代理:
		* 间接。
		* 获取代理对象:
		* 要求:
		* 被代理类最少实现一个接口
		* 创建的方式
		* Proxy.newProxyInstance(三个参数)
		* 参数含义:
		* ClassLoader:和被代理对象使用相同的类加载器。
		* Interfaces:和被代理对象具有相同的行为。实现相同的接口。
		* InvocationHandler:如何代理。
		* 
		*/
		//(IActor)Proxy.newProxyInstance,这里强制转换必须是接口类型
		IActor proxyActor = (IActor)Proxy.newProxyInstance(actor.getClass().getClassLoader(),
				 actor.getClass().getInterfaces(),
				 new InvocationHandler() {
			 /**
			 * 执行被代理对象的任何方法,都会经过该方法。
			 * 此方法有拦截的功能。
			 * 
			 * 参数:
			 * proxy:代理对象的引用。不一定每次都用得到
			 * method:当前执行的方法对象
			 * args:执行方法所需的参数
			 * 返回值:
			 * 当前执行被代理对象方法的返回值
			 */
					@Override
					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
						String name = method.getName();
						Float money = (Float) args[0];//执行的方法只有一个参数
						Object rtValue = null;
						//每个经纪公司对不同演出收费不一样,此处开始判断
						if("basicAct".equals(name)){
						//基本演出,没有 2000 不演
						if(money > 2000){
						//看上去剧组是给了 8000,实际到演员手里只有 4000
						//这就是我们没有修改原来 basicAct 方法源码,对方法进行了增强
						rtValue = method.invoke(actor, money/2);
						} }
						if("dangerAct".equals(name)){
						//危险演出,没有 5000 不演
						if(money > 5000){
						//看上去剧组是给了 50000,实际到演员手里只有 25000
						//这就是我们没有修改原来 dangerAct 方法源码,对方法进行了增强
						rtValue = method.invoke(actor, money/2);
						} }
						return rtValue;
					}
				});
		//没有经纪公司的时候,直接找演员。
		// actor.basicAct(1000f);
		// actor.dangerAct(5000f);
		//剧组无法直接联系演员,而是由经纪公司找的演员
		proxyActor.basicAct(2000f);//价格低于2000不演
		proxyActor.dangerAct(50000f);
		
	}
}

总结:

首先需要创建一个interface然后一个class实现这个interface,然后对这个class进行代理,这个class必须实现至少一个接口

基于子类的动态代理

设计的类:Enhancer
提供者:第三方cglib库
如何创建代理对象:
	使用Enhancer类中的create方法
创建代理对象的要求:
	被代理对象不是最终类(最终类没有子类)
create方法的参数:
	Class方法的参数:
		Class:字节码
			它是用于指定被代理对象的字节码
		callback:用于提供增强的代码
			它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。此接口的实现类都是谁用谁写。
			我们一般写的都是该接口的子接口实现类:MethodInterceptor
//CGLib动态代理(基于子类的动态代理)
Enhancer.create(两个参数);

代码如下:

package com.haust.serviceImpl;

public class Actor{//没有实现任何接口
	/**
	* 一个演员
	*/
	public void basicAct(float money) {
		System.out.println("拿到钱,开始基本的表演:"+money);
	}
	public void dangerAct(float money) {
		System.out.println("拿到钱,开始危险的表演:"+money);
	}
}
package com.haust.test;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import com.haust.serviceImpl.Actor;
public class test {
	public static void main(String[] args) {
		 Actor actor = new Actor();//需要创建此被代理的对象
		 /**
		 * 基于子类的动态代理
		 * 要求:
		 * 被代理对象不能是最终类
		 * 用到的类:
		 * Enhancer
		 * 用到的方法:
		 * create(Class, Callback)
		 * 方法的参数:
		 * Class:被代理对象的字节码
		 * Callback:如何代理
		 * @param args
		 */
		 //此时强转的类的类型就是被代理类的类型
		 Actor cglibActor = (Actor)Enhancer.create(actor.getClass(),new MethodInterceptor() {
			@Override
			public Object intercept(Object proxy, Method method, Object[] args, MethodProxy  methodProxy) throws Throwable {
				/**
				* 执行被代理对象的任何方法,都会经过该方法。在此方法内部就可以对被代理对象的任何
				方法进行增强。
				* 
				* 参数:
				* 前三个和基于接口的动态代理是一样的。
				* MethodProxy:当前执行方法的代理对象。
				* 返回值:
				* 当前执行方法的返回值
				*/
				String name = method.getName();
				Float money = (Float) args[0];
				Object rtValue = null;
				if("basicAct".equals(name)){
				//基本演出
				if(money > 2000){
				rtValue = method.invoke(actor, money/2);
				} }
				if("dangerAct".equals(name)){
					//危险演出
					if(money > 5000){
					rtValue = method.invoke(actor, money/2);
					} 
				}
				return rtValue;
			}
		});
		 cglibActor.basicAct(10000);
		 cglibActor.dangerAct(100000);
	}
}

总结:

无论哪种代理方式,都需要创建一个被代理的类(实例)。

不管是基于接口的代理,还是基于子类的代理,均拦截被代理对象的所有方法,然后我们可以对这些方法进行增强或者其他一些操作。

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • Springboot自动配置原理及DataSource的应用方式

    Springboot自动配置原理及DataSource的应用方式

    这篇文章主要介绍了Springboot自动配置原理及DataSource的应用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • MyBatis结果映射(ResultMap)的使用

    MyBatis结果映射(ResultMap)的使用

    在MyBatis中,结果映射是实现数据库结果集到Java对象映射的核心,它不仅支持简单的字段映射,还能处理字段名不一致、嵌套对象和集合映射等复杂场景,通过ResultMap,开发者可以灵活定义映射关系,以适应各种需求,感兴趣的可以了解一下
    2024-09-09
  • idea类名显示多行的设置方式

    idea类名显示多行的设置方式

    在IntelliJ IDEA中,类名的显示方式可以通过设置来调整,若想设置为单行显示,需在设置中找到相关选项并勾选“√”,若需多行显示,则取消勾选即可,此操作有助于优化代码视图,提升开发效率
    2024-09-09
  • Java如何基于反射获取对象属性信息

    Java如何基于反射获取对象属性信息

    这篇文章主要介绍了Java如何基于反射获取对象属性信息,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • Java中的对象和对象引用实例浅析

    Java中的对象和对象引用实例浅析

    这篇文章主要介绍了Java中的对象和对象引用,实例分析了对象与对象引用的概念与相关使用技巧,需要的朋友可以参考下
    2015-05-05
  • Spring Security权限注解启动及逻辑处理使用示例

    Spring Security权限注解启动及逻辑处理使用示例

    这篇文章主要为大家介绍了Spring Security权限注解启动及逻辑处理使用示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-07-07
  • Java版微信公众号支付开发全过程

    Java版微信公众号支付开发全过程

    这篇文章主要介绍了Java版微信公众号支付开发全过程,本文通过实例相结合给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-07-07
  • spring boot idea maven依赖找不到问题处理方法

    spring boot idea maven依赖找不到问题处理方法

    这篇文章主要介绍了spring boot idea 偶尔maven依赖找不到问题,这里总结了几种处理方法,方便尝试排查,对spring boot idea  maven依赖找不到问题感兴趣的朋友跟随小编一起看看吧
    2023-08-08
  • SpringBoot中@Import注解如何正确使用

    SpringBoot中@Import注解如何正确使用

    这篇文章主要介绍了SpringBoot中@Import注解的使用方式,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-06-06
  • IDEA删除的文件怎么找回更新的方法

    IDEA删除的文件怎么找回更新的方法

    查找本地历史记录IDEA在进行代码版本管理时,会自动创建本地历史记录,如果我们误删了文件,可以通过查找本地历史记录来找回文件,本文就来介绍一下
    2023-11-11

最新评论