动态代理原理

一、案例分析

 1、引出问题

回到Spring之初控制事务繁琐的问题。 回到Spring之初控制事务繁琐的问题.

考虑一个应用场景∶需要对系统中的某些业务方法做事务管理,拿简单的save和update操作举例。没有加上事务控制的代码如下。

加上事务代码,如下:

上述问题︰在我们的业务层中每一个业务方法都得处理事务(繁琐的try-catch)

在设计上存在两个很严重问题︰ 上述问题:在我们的业务层中每一个业务方法都得处理事务(繁琐的尝试-捕捉)。在设计上存在两个很严重问题:

责任不分离.业务方法只需要关心如何完成该业务功能,不需要去关系事务管理/日志管理/权限管理等等。

代码结构重复.在开发中不要重复代码,重复就意味着维护成本增大。

2、房屋租赁的启示

通过中介,房东就可以免去一系列繁琐的东西,只管收租就好了

 二、静态代理

1、代理实现

 代理模式︰客户端直接使用的都是代理对象,不知道真实对象是谁,此时代理对象可以在客户端和真实对象之间起到中介的作用。

1、代理对象完全包含真实对象,客户端使用的都是代理对象的方法,和真实对象没有直接关系;

2、代理模式的职责︰把不是真实对象该做的事情从真实对象上撇开——职责清晰;

静态代理∶在程序运行前就已经存在代理类的字节码文件,代理对象和真实对象的关系在运行前就确定了。

EmployeeServiceProxy 代码:

//静态代理类
public class EmployeeServiceProxy implements IEmployeeService {
	private IEmployeeService target;//真实对象/委托对象

	private TransactionManager txManager;//事务管理器

	public void setTarget(IEmployeeService target) {
		this.target = target;
	}

	public void setTxManager(TransactionManager txManager) {
		this.txManager = txManager;
	}

	public void save(Employee emp) {
		txManager.begin();
		try {
			target.save(emp);
			txManager.commit();
		} catch (Exception e) {
			e.printStackTrace();
			txManager.rollback();
		}
	} 
}

XML如下

	<bean id="employeeDAO" class="cn.wolfcode.dao.impl.EmployeeDAOImpl" />

	<bean id="transactionManager" class="cn.wolfcode.tx.TransactionManager" />
	
	<!-- 代理对象 -->
	<bean id="employeeServiceProxy" class="cn.wolfcode.proxy.EmployeeServiceProxy">
		<property name="txManager" ref="transactionManager" />
		<property name="target">
			<bean class="cn.wolfcode.service.EmployeeServiceImpl">
				<property name="dao" ref="employeeDAO" />
			</bean>
		</property>
	</bean>

调用

	@Test
	void testSave() throws Exception {
		System.out.println(service.getClass());//查看对象的真实类型
		service.save(new Employee());
	}

2、问题分析

静态代理∶在程序运行前就已经存在代理类的字节码文件,代理对象和真实对象的关系在运行前就确定了。

优点︰
① 业务类只需要关注业务逻辑本身,保证了业务类的重用性。

② 把真实对象隐藏起来了,保护真实对象

缺点︰

① 代理对象的某个接口只服务于某一种类型的对象,也就是说每一个真实对象都得创建一个代理对象

② 如果需要代理的方法很多,则要为每一种方法都进行代理处理。

③ 如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。
 

三、动态代理

代理模式︰客户端直接使用的都是代理对象,不知道真实对象是谁,此时代理对象可以在客户端和真实对象之间起到中介的作用。

1、代理对象完全包含真实对象,客户端使用的都是代理对象的方法,和真实对象没有直接关系;

2、代理模式的职责:把不是真实对象该做的事情从真实对象上撇开——职责清晰;

静态代理∶在程序运行前就已经存在代理类的字节码文件,代理对象和真实对象的关系在运行前就确定了。

动态代理∶动态代理类是在程序运行期间由JVM通过反射等机制动态的生成的,所以不存在代理类的字节码文件,代理对象和真实对象的关系是在程序运行时期才确定的。

如何实现动态代理∶

1)︰针对有接口∶使用JDK动态代理

2)∶针对无接口∶使用CGLIB或Javassist组件

1、字节码动态加载

Java运行原理和字节码加载过程:

 如何动态的加载一份字节码︰

由于JVM通过字节码的二进制信息加载类的,如果我们在运行期系统中,遵循Java编译系统组织.class文件的格式和结构,生成相应的二进制数据,然后再把这个二进制数据加载转换成对应的类,如此,就完成了在代码中动态创建一个类的能力了。

2、JDK动态代理

JDK动态代理API分析:(必须要求真实对象是有接口)
1) java.lang.reflect.Proxy类:Java动态代理机制生成的所有动态代理类的父类,它提供了一组静态方法来为一组接口动态地生成代理类及其对象。
主要方法∶

public static Object newProxyInstance(ClassLoader loader, Class<?>[]interfaces,InvocationHandler handler)

方法职责∶为指定类加载器、一组接口及调用处理器生成动态代理类实例

loader:类加载器,一般传递真实对象的类加载器
interfaces:代理类需要实现的接口
hanlder:代理对象如何做增强

2)java.lang.reflect.InvocationHandler接口:

public Object invoke(Object proxy, Method method, Object[] args)

方法职责︰负责集中处理动态代理类上的所有方法调用参数:
proxy :生成的代理对象
method :当前调用的真实方法对象
args:当前调用方法的实参返回:真实方法的返回结果

jdk动态代理操作步骤∶
①实现InvocationHandler接口,创建自己增强代码的处理器。
②给Proxy类提供ClassLoader对象和代理接口类型数组,创建动态代理对象。
③在处理器中实现增强操作。

Java代码:

//事务的增强操作
public class TransactionManagerAdvice implements java.lang.reflect.InvocationHandler {

	private Object target;//真实对象(对谁做增强)
	private TransactionManager txManager;//事务管理器(模拟)

	public void setTxManager(TransactionManager txManager) {
		this.txManager = txManager;
	}

	public void setTarget(Object target) {
		this.target = target;
	}

	//创建一个代理对象
	public <T> T getProxyObject() {
		return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), //类加载器,一般跟上真实对象的类加载器
				target.getClass().getInterfaces(), //真实对象所实现的接口(JDK动态代理必须要求真实对象有接口)
				this);//如何做事务增强的对象
	}

	//如何为真实对象的方法做增强的具体操作
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		if (method.getName().startsWith("get") || method.getName().startsWith("list")) {
			return method.invoke(target, args);//放行
		}
		
		Object ret = null;
		txManager.begin();
		try {
			//---------------------------------------------------------------
			ret = method.invoke(target, args);//调用真实对象的方法
			//---------------------------------------------------------------
			txManager.commit();
		} catch (Exception e) {
			e.printStackTrace();
			txManager.rollback();
		}
		return ret;
	}
}

3、JDK动态代理原理

通过DynamicProxyClassGenerator生成动态代理的字节码,再通过反编译工具查看。

生成动态代理字节码。

public class DynamicProxyClassGenerator {
	public static void main(String[] args) throws Exception {
		generateClassFile(EmployeeServiceImpl.class, "EmployeeServiceProxy");
	}

	//生成代理类的字节码文件-->Java反编译工具-->Java文件
	public static void generateClassFile(Class targetClass, String proxyName) throws Exception {
		//根据类信息和提供的代理类名称,生成字节码  
		byte[] classFile = ProxyGenerator.generateProxyClass(proxyName, targetClass.getInterfaces());
		String path = targetClass.getResource(".").getPath();
		System.out.println(path);
		FileOutputStream out = null;
		//保留到硬盘中  
		out = new FileOutputStream(path + proxyName + ".class");
		out.write(classFile);
		out.close();
	}

}

生成后代码(整理过)

public final class EmployeeServiceProxy extends Proxy implements IEmployeeService {
	private static Method method_equals;
	private static Method method_toString;
	private static Method method_hashCode;
	private static Method method_update;
	private static Method method_save;

	public EmployeeServiceProxy(InvocationHandler paramInvocationHandler) {
		super(paramInvocationHandler);
	}
	
	static {
		try {
			method_equals = Class.forName("java.lang.Object").getMethod("equals",new Class[] { Class.forName("java.lang.Object") });
			method_toString = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
			method_hashCode = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
			
			method_update = Class.forName("cn.wolfcode.service.IEmployeeService").getMethod("update",new Class[] { Class.forName("cn.wolfcode.domain.Employee") });
			method_save = Class.forName("cn.wolfcode.service.IEmployeeService").getMethod("save",new Class[] { Class.forName("cn.wolfcode.domain.Employee") });
		} catch (Exception e) {
		} 
	}

	public final boolean equals(Object paramObject) {
		try {
			return ((Boolean) this.h.invoke(this, method_equals, new Object[] { paramObject })).booleanValue();
		} catch (Error | RuntimeException localError) {
			throw localError;
		} catch (Throwable localThrowable) {
			throw new UndeclaredThrowableException(localThrowable);
		}
	}

	

	public final String toString() {
		try {
			return (String) this.h.invoke(this, method_toString, null);
		} catch (Error | RuntimeException localError) {
			throw localError;
		} catch (Throwable localThrowable) {
			throw new UndeclaredThrowableException(localThrowable);
		}
	}

	public final int hashCode() {
		try {
			return ((Integer) this.h.invoke(this, method_hashCode, null)).intValue();
		} catch (Error | RuntimeException localError) {
			throw localError;
		} catch (Throwable localThrowable) {
			throw new UndeclaredThrowableException(localThrowable);
		}
	}

	public final void save(Employee paramEmployee) {
		try {
			this.h.invoke(this, method_save, new Object[] { paramEmployee });
			return;
		} catch (Error | RuntimeException localError) {
			throw localError;
		} catch (Throwable localThrowable) {
			throw new UndeclaredThrowableException(localThrowable);
		}
	}
	
	public final void update(Employee paramEmployee) {
		try {
			this.h.invoke(this, method_update, new Object[] { paramEmployee });
			return;
		} catch (Error | RuntimeException localError) {
			throw localError;
		} catch (Throwable localThrowable) {
			throw new UndeclaredThrowableException(localThrowable);
		}
	}

}

原理一图胜千言

 观察: save方法,发现底层其实依然在执行InvocationHandler中的invoke方法。

注意:在增强的方法里面调用toString会导致死循环(原理看生成的代理类源码)

4、CGLIB动态代理

使用JDK的动态代理,只能针对于目标对象存在接口的情况,如果目标对象没有接口,此时可以考虑使用CGLIB 的动态代理方式。
Java 代码︰

//事务的增强操作-CGLIB
public class TransactionManagerAdvice implements org.springframework.cglib.proxy.InvocationHandler {

	private Object target;//真实对象(对谁做增强)
	private TransactionManager txManager;//事务管理器(模拟)

	public void setTxManager(TransactionManager txManager) {
		this.txManager = txManager;
	}

	public void setTarget(Object target) {
		this.target = target;
	}

	//创建一个代理对象
	public <T> T getProxyObject() {
		Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(target.getClass());//将继承于哪一个类,去做增强
		enhancer.setCallback(this);//设置增强的对象
		return (T) enhancer.create();//创建代理对象
	}

	//如何为真实对象的方法做增强的具体操作
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		Object ret = null;
		txManager.begin();
		try {
			//---------------------------------------------------------------
			ret = method.invoke(target, args);//调用真实对象的方法
			//---------------------------------------------------------------
			txManager.commit();
		} catch (Exception e) {
			e.printStackTrace();
			txManager.rollback();
		}
		return ret;
	}
}

4、CGLIB动态代理原理

System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,"C:/test");
CGLIB生成的动态代理字节码文件,阅读比较复杂,不容易读,我们见到看一下,下面是经过优化处理的代码。

观察∶可以看出CGLIB是通过生成代理类,然后继承于目标类,再对目标类中可以继承的方法做覆盖,并在该方法中做功能增强的,因为多态的关系,实则调用的是子类中的方法。
 

四、动态代理总结

1、JDK动态代理总结︰

① JAVA动态代理是使用java.lang.reflect包中的Proxy类与InvocationHandler接口这两个来完成的。

② 要使用JDK动态代理,委托必须要定义接口

③ JDK动态代理将会拦截所有pubic的方法(因为只能调用接口中定义的方法),这样即使在接口中增加了新的方法,不用修改代码也会被拦截。

④ 动态代理的最小单位是类(所有类中的方法都会被处理),如果只想拦截一部分方法,可以在invoke方法中对要执行的方法名进行判断。

2、CGLIB代理总结∶

① CGLIB可以生成委托类的子类,并重写父类非final修饰符的方法。

② 要求类不能是final的,要拦截的方法要是非final、非static、非private的。

③ 动态代理的最小单位是类(所有类中的方法都会被处理);
 

备注:如果想深入彻底理解底层代码,可以在评论区发获取动态代理原理解刨资料哦

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/6671.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大数据平台开发——使用Java和Python调用Shell脚本

大数据平台开发——使用Java和Python调用Shell脚本 背景 在大数据平台开发中&#xff0c;经常会遇到需要调用Shell脚本的场景&#xff0c;倒不是说只能用Shell&#xff0c;毕竟大数据开发到头来一定是个语言无关的事情&#xff1a; 从Hive源码解读大数据开发为什么可以脱离S…

Java进阶

注解 什么是注解 Java注解&#xff08;Annotation&#xff09;又称Java标注&#xff0c;是JDK5.0引入的一种注释机制。 Java语言中类、方法、变量、参数和包等都可以被标注。Java标注可以通过反射获取标注内容。在编译器生成类文件时&#xff0c;标注可以被嵌入到字节码中。Ja…

【Python实操】一行代码就可以自动画出这种艺术画?(详细教程)

文章目录前言一.准备阶段二、开始使用 Discoart1.引入库2.显示/保存/加载配置总结前言 DiscoArt 是一个很牛逼的开源模块&#xff0c;它能根据你给定的关键词自动绘画。 绘制过程是完全可见的&#xff0c;你可以在 jupyter 页面上看见这个绘制的过程&#xff1a; 一.准备阶段…

零拷贝内存 固定内存

一、总览 虚拟内存是一种计算机内存管理的技术&#xff0c;它让程序认为程序自身有一段完整的连续可用的内存&#xff08;一个地址空间&#xff09;。当程序运行时所占的内存空间大于物理空间容量&#xff0c;操作系统可以将暂时不用的数据放入到磁盘&#xff0c;用的时候再拿出…

Linux--高级IO--select--0326

目录 IO为什么低效&#xff1f; 1.快速理解五种IO模式 2.五种IO模型 3.非阻塞IO fcntl() 4.IO多路转接 select select fd_set类型 struct timeval*类型 5.Select的代码测试 5.1 问题一&#xff1a;一开始&#xff0c;我们只有一个listen套接字 5.2 问题二&#xff1…

《项目管理知识体系指南(PMBOK)》第7版之8大绩效域

项目绩效域被定义为一组对有效交付项目成果至关重要的相关活动。 《项目管理知识体系指南&#xff08;PMBOK&#xff09;》第7版将项目管理划分为干系人、团队、开发方法和生命周期、规划、项目工作、交付、测量、不确定性共8大绩效域。 一、干系人绩效域 解决与干系人相关的…

【对YOLOv8(ultralytics)打印测试结果的调整】(1)使得map值打印显示从0.551变为55.08 (2)打印出FPS

目录1. 最终打印效果2. 做两处更改2.1 修改map显示&#xff0c;在ultralytics-main/ultralytics/yolo/v8/detect/val.py中操作2.2 打印FPS&#xff0c;在ultralytics-main/ultralytics/yolo/engine/validator.py中操作❗❗❗ 兄弟姐妹们&#xff0c;如果看习惯了运行train.py时…

PMP应该如何备考?

PMP现在是新考纲&#xff0c;PMP新版大纲加入了 ACP 敏捷管理的内容&#xff0c;而且还不少&#xff0c;敏捷混合题型占到了 50%&#xff0c;前不久官方也发了通知 8月启用第七版《PMBOK》&#xff0c;大家都觉得考试难度提升了&#xff0c;我从新考纲考完下来&#xff0c;最开…

Moonbeam隆重推出您的个人开发小助手 — — Kapa.ai

Moonbeam为开发者提供内容详细的开发者文档和全天候的Discord支持。但假如&#xff1a;有人可以24/7查看Discord并在15秒之内就回复您的问题 — — 新推出的Kapa.ai机器人使这个假如成为现实。Kapa.ai是一款由ChatGPT支持的AI机器人&#xff0c;可以回答关于在Moonbeam上构建的…

【redis】单线程redis为什么这么快

本文以收录专栏 redis核心技术 前言 本专栏为了帮助大家更好的了解学习redis&#xff0c;同时也是自己记录学习redis的内容&#xff0c;包含了大部分的redis核心技术&#xff0c;分布式锁&#xff0c;主从复制等 目录 专题2-单线程redis为什么这么快 2.1redis只有单线程吗&a…

剑指offer-替换空格

替换空格一、解题思想二、代码的实现三、总结一、解题思想 题目&#xff1a;请实现一个函数 &#xff0c;把字符串中的每个空格替换成”%20“。例如&#xff1a;输入”We are happy.“&#xff0c;则输出”We%20are%20happy.“。 看到这个题目&#xff0c;我第一想到的是&#…

博客1:YOLOv5车牌识别实战教程:引言与准备工作

摘要:本篇博客介绍了本教程的目标、适用人群、YOLOv5简介和车牌识别的意义和应用场景。为后续章节打下基础,帮助读者了解YOLOv5和车牌识别的相关背景知识。 正文: 车牌识别视频 引言 欢迎来到YOLOv5车牌识别实战教程!在这个教程中,我们将一步步教你如何使用YOLOv5进行车…

【Git Bash】项目开发过程中需要知道 git stash 的用法

目录1. git stash的应用场景2. 常用git stash命令2.1 git stash2.2 git stash save "message"2.3 git stash list2.4 git stash show2.5 git stash show -p2.6 git stash apply2.7 git stash pop2.8 git stash drop stash{num}2.9 git stash clear3. stash只会保存已…

简单记录一下软著申请流程

模板我就不放了&#xff0c;网上很多&#xff0c;随便下几个结合就行了 总体来说&#xff0c;我是2023.2.19号寄出&#xff0c;2023.4.6看到成功了&#xff0c;总共50天左右。 大家确实不需要网上买那种服务&#xff0c;我也是第一次申请&#xff0c;感觉还是挺简单的。只要按…

洛谷B2038奇偶ASCII值判断

洛谷B2038 题目描述 任意输入一个字符&#xff0c;判断其 ASCII 是否是奇数&#xff0c;若是&#xff0c;输出 YES&#xff0c;否则&#xff0c;输出 NO 。 例如&#xff0c;字符 A 的 ASCII 值是 65&#xff0c;则输出 YES&#xff0c;若输入字符 B(ASCII 值是 66)&#xff0…

从零开始学习Kotlin,带你快速掌握该编程语言

前言 Kotlin是一种跨平台的静态编程语言&#xff0c;它可以在JVM、Android、浏览器、iOS等多个平台上运行。Kotlin的语法简洁易懂&#xff0c;具有高度的可读性和可维护性&#xff0c;同时还具有Java所不具备的许多优点。 Kotlin是一种静态类型、面向对象、函数式编程语言&am…

iOS 项目嵌入Flutter 运行

一 创建Flutter 模块命令行flutter create --template module my_flutter创建完成后&#xff0c;该模块和普通的Flutter项目一直&#xff0c;可以通过Android Studio或VSCode打开、开发、运行&#xff1b;和之前项目不同的iOS和Android项目是一个隐藏文件&#xff0c;并且我们…

多模态 |COGMEN: COntextualized GNN based Multimodal Emotion recognitioN论文详解

论文&#xff1a;COGMEN: COntextualized GNN based Multimodal Emotion recognitioN COGMEN: 基于GNN的多模态情感识别技术 论文实现可参考另外一篇论文&#xff1a; 本文主要分为俩部分&#xff0c;一是对论文的简单概括&#xff0c;二是对论文的翻译。 论文总结 论文翻译…

【学习笔记】SpringAOP的用法全解

文章目录Spring的AOP一、 Spring对AOP的实现包括以下3种方式**什么是AspectJ?**二、使用Spring的AOP1、准备工作2、尝试写一个简单的AOP demo3、代码如下&#xff1a;spring.xml业务类切面类测试类4、复习切面表达式1&#xff09;所有方法2&#xff09;指定路径下某个包及其子…

开心档之C++ 运算符

目录 C 运算符 算术运算符 实例 实例 关系运算符 实例 实例 逻辑运算符 实例 实例 位运算符 实例 实例 赋值运算符 实例 实例 杂项运算符 C 中的运算符优先级 实例 实例 运算符是一种告诉编译器执行特定的数学或逻辑操作的符号。C 内置了丰富的运算符&…