【Spring Boot】Spring AOP动态代理,以及静态代理

目录

  • Spring AOP代理
    • 一. 代理的概念
    • 二. 静态代理
    • 三. JDK代理
      • 3.1 重写 invoke 方法进⾏功能增强
      • 3.2 通过Proxy类随机生成代理对象
    • 四. CGLIB代理
      • 4.1 自定义类来重写intercept方法
      • 4.2 通过Enhancer类的create方法来创建代理类
    • 五. AOP源码剖析
  • 总结(重中之重,精华)

Spring AOP代理

一. 代理的概念

根据前面的学习想必大家都已经对Spring AOP有所了解了,接下来我们先来回忆一下什么是Spring AOP?

AOP:一种对于集中的事情进行统一处理解决的思想;

Spring AOP:Spring通过运用AOP统一解决的思想所诞生的产物;
例如:拦截器,适配器,统一结果返回,统一异常处理,以及统一通知处理,以上这些在我们前面的文章中都讲述过,已经有些遗忘的小伙伴可以翻看前面的文章进行稳固一下…

好!接下来我们进入正题…

AOP的底层原理实现的代理模式,那么什么是代理模式呢?
通过举一个栗子~大家就应该能够了解了:有些 小伙伴可能通过一些线上平台租过房子,那么这个线上平台就是我们说的中介,是在我们跟房子房东之间的纽带,为啥我们要找中介呢?

  1. 中介能够帮我们提前去验收要出租房子的质量来进行出租价格的评定
  2. 中介能够在我们住房期间能够对房子的进行一个改造升级

通过上面的栗子中的 中介就是代理,我们通过中介能够达到我们最终的要求,这就是代理模式简单来说就是通过一个代理类能够间接的调用目标方法
在这里插入图片描述

然而代理模式又分为两种:

  1. 静态代理
  2. 动态代理(两种)

静态代理和动态代理的主要区别就是:静态代理的代理对象的一开始就定好的,而动态代理就跟他的命名一样,是动态化的,是由系统随机调度生成的一个代理对象
按照上面的例子来说就是,静态代理A的房子,那么A房子的中介人一直是这个人,而动态代理是中介公司看现在哪一个中介在摸鱼,就让哪一个中介去干活~
当然了,在面试中主要考查的是动态代理;

动态代理(主要是通过反射来完成的代理模式)

  1. JDK代理
  2. CGLIB代理

在接下来的讲解中我们将围绕以上几种代理进行展开,由于JDK代理和CGLIB代理是面试中的重中之重,篇幅较长,我们后面慢慢讲述,先就简单的,软的柿子——静态代理来捏~~

二. 静态代理

静态代理:由程序员创建代理类或特定⼯具⾃动⽣成源代码再对其编译,在程序运⾏前代理类的
.class⽂件就已经存在了
什么意思呢?简单来理解就是它的代理对象已经定死了,不会在修改了。

代理(中介,帮房东出租房⼦)

public class HouseProxy implements HouseSubject{

	//将被代理对象声明为成员变量
	private HouseSubject houseSubject;
	
	public HouseProxy(HouseSubject houseSubject) {
		this.houseSubject = houseSubject;
	}
	
	@Override
	public void rentHouse() {
	
		//开始代理
		System.out.println("我是中介, 开始代理");
		
		//代理房东出租房⼦
		houseSubject.rentHouse();
		
		//代理结束
		System.out.println("我是中介, 代理结束");
	}
}

这段代码不需要看明白,只用记住,代理对象通过renHouse方法来加强房子的质量,以及代理对象写死了就行了,重点全在动态代理中

三. JDK代理

在上面讲过,动态代理于静态代理的区别在于动态代理的代理对象是系统生成的,而JDK动态代理是通过:
JDK动态代理有⼀个最致命的问题是其只能代理实现了接口的类

  1. 通过实现InvocationHandler接口,重写 invoke 方法来加强被代理的对象(加强房子的质量)
  2. 通过Proxy类来随机生成一个代理对象

3.1 重写 invoke 方法进⾏功能增强

实现InvocationHandler接口,重写的 invoke 方法的伪代码如下,了解他们的区别和调用的方法即可:

public interface InvocationHandler {
/**
* 参数说明
* proxy:代理对象
* method:代理对象需要实现的方法,即其中需要重写的方法
* args:method所对应方法的参数
*/
public Object invoke(Object proxy, Method method, Object[] args)
	throws Throwable;
}

InvocationHandler接口是Java动态代理的关键接口之⼀,它定义了⼀个单⼀方法 invoke() ,⽤于
处理被代理对象的方法调⽤.通过实现 InvocationHandler 接口,可以对被代理对象的方法进⾏功能增强.

3.2 通过Proxy类随机生成代理对象

通过Proxy类来随机生成一个代理对象伪代码如下:

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
throws IllegalArgumentException
{
//...代码省略
}

Proxy 类中使⽤频率最⾼的方法是: newProxyInstance() ,这个方法主要⽤来⽣成⼀个代理
对象

这个方法⼀共有3个参数:
Loader:类加载器,⽤于加载代理对象.
interfaces:被代理类实现的⼀些接口(这个参数的定义,也决定了JDK动态代理只能代理实现了接口的
⼀些类)
h:实现了InvocationHandler接口的对象

⾃定义 InvocationHandler 并重写 invoke 方法,在 invoke 方法中我们会调⽤⽬标方
法(被代理类的方法)并⾃定义⼀些处理逻辑
3. 通过 Proxy.newProxyInstance(ClassLoader loader,Class<?>[]
interfaces,InvocationHandler h) 方法创建代理对象

四. CGLIB代理

JDK动态代理有⼀个最致命的问题是其只能代理实现了接口的类,而CGLIB却能够实现接口也能代理类

  1. 通过⾃定义 MethodInterceptor 并重写 intercept 方法加强被代理的对象(加强房子的质量)
  2. 通过Enhancer类的create()创建代理类

和JDK动态代理不同,CGLIB(CodeGenerationLibrary)实际是属于⼀个开源项⽬,如果你要使⽤它
的话,需要⼿动添加相关依赖

<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.3.0</version>
</dependency>

4.1 自定义类来重写intercept方法

⾃定义 MethodInterceptor 并重写 intercept 方法, intercept ⽤于增强⽬标方法,和JDK动态代理中的 invoke 方法类似

代码如下:

import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;

public class CGLIBInterceptor implements MethodInterceptor {
	
		//⽬标对象, 即被代理对象
	private Object target;	
	
	public CGLIBInterceptor(Object target){
		this.target = target;
	}
	
	@Override
	public Object intercept(Object o, Method method, Object[] objects,MethodProxy methodProxy) throws Throwable {
	
		// 代理增强内容
		System.out.println("我是中介, 开始代理");
		
		//通过反射调⽤被代理类的方法
		Object retVal = methodProxy.invoke(target, objects);
		
		//代理增强内容
		System.out.println("我是中介, 代理结束");
		
		return retVal;
	}
}

MethodInterceptor 和JDK动态代理中的 InvocationHandler 类似,它只定义了⼀个方
法 intercept() ,⽤于增强⽬标方法

4.2 通过Enhancer类的create方法来创建代理类

通过Enhancer类的create()创建代理类

伪代码如下:

public static Object create(Class type, Callback callback) {
//...代码省略
}

参数说明:
type:被代理类的类型(类或接口)
callback:⾃定义方法拦截器MethodInterceptor

五. AOP源码剖析

SpringAOP主要基于两种方式实现的:JDK及CGLIB的方式

然而为什么会有两种不同的代理方式?以及什么时候用不同的代码方式呢?
在下面的部分源码中我们可以了解到

//创建代理⼯⼚
ProxyFactory proxyFactory = new ProxyFactory();
	proxyFactory.copyFrom(this);

/**
* 检查proxyTargetClass属性值,spring默认为false
* proxyTargetClass 检查接口是否对类代理, ⽽不是对接口代理
* 如果代理对象为类, 设置为true, 使⽤cglib代理
*/
if (!proxyFactory.isProxyTargetClass()) {

	//是否有设置cglib代理
	if (shouldProxyTargetClass(beanClass, beanName)) {
	
		//设置proxyTargetClass为true,使⽤cglib代理
		proxyFactory.setProxyTargetClass(true);
	} else {
	
		/**
		* 如果beanClass实现了接口,且接口⾄少有⼀个⾃定义方法,则使⽤JDK代理
		* 否则CGLIB代理(设置ProxyTargetClass为true )
		* 即使我们配置了proxyTargetClass=false, 经过这⾥的⼀些判断还是可能会将其
		设为true
		*/
		evaluateProxyInterfaces(beanClass, proxyFactory);
	}
}

从上面的部分源码来看,接下来是我自己用大白话总结出来的道理,
首先会创建一个代理工厂,用于创建两种不同的代理对象,然后源码中有一个proxyTagertClass属性来进行判断,如果proxyTagertClass为false,且实现了接口,则用代理工厂来创建一个JDK代理的对象,如果没有实现接口或者proxyTagertClass为true,则用代理工厂来创建一个CGLIB代理的对象

根据proxyTagertClass属性来创建代理对象的情况如下表格:

proxyTargetClass⽬标对象代理方式
false实现了接口jdk代理
false未实现接口(只有实现类)cglib代理
true实现了接口cglib代理
true未实现接口(只有实现类)cglib代理
注意:这里的版本不同,创建的方式也会不同,在Spring版本中proxyTagertClass属性默认的为false,然而在Spring Boot 2.0版本以后proxyTagertClass属性则默认的是true ,也就是默认使用CGLIB来代理

总结(重中之重,精华)

两大重要的动态代理:

  1. JDK动态代理
  2. CGLIB动态代理

以及它两个的区别:

  1. JDK代理主要用于实现接口,不能直接用于类;而CGLIB在实现接口和类上都可以使用
  2. JDK代理通过重写invoke方法来进行功能增强,通过proxy类来生成随机代理对象,而CGLIB代理通过自定义类来重写intercept方法,通过Enhancer类的create方法来创建代理类

从动态代理的部分源码来看,接下来是我自己用大白话总结出来的过程,
首先会创建一个代理工厂,用于创建两种不同的代理对象,然后源码中有一个proxyTagertClass属性来进行判断,如果proxyTagertClass为false,且实现了接口,则用代理工厂来创建一个JDK代理的对象,如果没有实现接口或者proxyTagertClass为true,则用代理工厂来创建一个CGLIB代理的对象

以及proxyTagertClass属性在Spring Boot 2.0版本之前默认是false,而在2.0版本以后默认为true

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

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

相关文章

一款简单、免费的web文件共享服务器

#共享文件# #远程访问# #手机访问# 文件共享已成为我们日常生活和工作中不可或缺的一部分。它如同一条无形的纽带&#xff0c;将人们紧密地联系在一起&#xff0c;促进了信息的快速传播和交流。 文件共享的魅力在于其打破了地域和时间的限制。无论我们身处世界的哪个角落&…

深度解析:当下流行的人工智能大模型生成逻辑

在过去的几年里&#xff0c;人工智能领域经历了前所未有的革新&#xff0c;其中最引人注目的就是大规模预训练模型的崛起。这些模型&#xff0c;如GPT系列、BERT、T5、DALLE和CLIP等&#xff0c;凭借其强大的语言理解和生成能力&#xff0c;已经在自然语言处理&#xff08;NLP&…

Springboot使用WebSocket发送消息

1. 创建springboot项目&#xff0c;引入spring-boot-starter-websocket依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency>完整项目依赖 <?xml ver…

聊聊使用GROUP_CONCAT函数遇到的坑

问题现象 在工作中我们或多或少都会使用到函数group_concat&#xff0c;它可以合并多行的某列(或多列)数据为一行&#xff0c;默认以逗号分隔。 最近碰到了一个线上bug&#xff0c;查询DB时返回的结果信息mysql自动截取了&#xff0c;导致页面显示的时候只显示了前半段结果。 …

MATLAB环境下4种噪声生成

生成噪声包括: 1)粉红色(闪烁)噪声-功率谱密度斜率-3 dB/oct。&#xff0c; - 10db /dec 2)红色(布朗)噪声-功率谱密度斜率-6 dB/oct。&#xff0c; - 20db /dec 3)蓝色噪声-功率谱密度斜率3 dB/oct。&#xff0c; 10db /dec 4)紫色(紫色)噪声-功率谱密度斜率 6db /oct。&…

抖音商城自定义小程序源码系统 前后端分离 带完整的源代码包以及搭建教程

系统概述 在当今数字化时代&#xff0c;电商平台的便捷性和个性化体验成为了吸引用户的关键。随着短视频平台的兴起&#xff0c;抖音作为其中的佼佼者&#xff0c;其商城小程序成为了商家连接消费者的新阵地。为了帮助商家快速构建个性化、高效的小程序店铺&#xff0c;本文将…

Java线程的创建·启动和休眠

一.线程的创建和启动 Java中创建线程的两种方式 ◆继承java.lang.Thread类 ◆实现java.lang.Runnable接口 ◆使用线程的步骤 继承Thread类创建线程 ◆自定义线程类继承自Thread类 ◆重写run()方法&#xff0c;编写线程执行体 ◆创建线程对象&#xff0c;调用start()方法启动…

基于大数据的电商产品评论数据分析与可视化--Python

基于大数据的电商产品评论数据分析与可视化 1绪论 1.1研究背景与意义阐述 随着电子商务领域的迅猛扩张,电商平台累积了海量的用户评价信息。这些建议不只是包含了消费者对产品的评价和经验分享,更重要的是,它们包含了丰富且价值巨大的信息。深度分析在线用户反馈不仅揭示…

#数据结构 链表

单向链表 1. 概念 单向链表 单向循环链表 双向链表 双向循环链表 解决&#xff1a;长度固定的问题&#xff0c;插入和删除麻烦的问题 1、逻辑结构&#xff1a; 线性结构 2、存储结构&#xff1a; 链式存储 链表就是将 结点 用链串起来的线性表&#xff0c;链就是 结点 中的…

《C++20设计模式》命令模式思考

文章目录 一、前言二、分析 拆解1、经典命令模式2、撤销操作3、关于Invoker类 三、实现 一、前言 哎&#xff01;只要是书上写的和经典设计模式不同&#xff0c;我就会很伤脑筋。&#x1f629; 命令模式到底是干什么的&#xff1f; 答&#xff1a;命令的发送者和接收者完全解…

环境配置05——conda创建虚拟环境指定版本torch与python

版本选择&#xff1a; python版本3.11.8torch版本2.1.2 1.创建环境 conda create -n t212p311 python3.11.8 2.下载torch pytorch-wheels-cu121安装包下载_开源镜像站-阿里云 3. 安装torch 进入虚拟环境 activate t212p311 进入torch安装包所在目录&#xff0c;安装torc…

html+css+js随机验证码

随机画入字符、线条 源代码在图片后面 点赞❤️关注&#x1f60d;收藏⭐️ 互粉必回 图示 源代码 <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"…

将QComboBox下拉项中的文本居中、居右

目录 1. 需求提出 2. 解决方法 1. 需求提出 QComboBox下拉项中的文本默认是居左的&#xff0c;如下&#xff1a; 有时需要将下拉项中的文本居中、居右。如何实现&#xff1f; 2. 解决方法 首先想到的是通过样式表来解决&#xff0c;但找遍Qt Assist和网络&#xff0c;都没这…

MySQL存储与优化 一、MySQL架构原理

1.MySQL体系架构 MySQL Server架构自顶向下大致可以分网络连接层、服务层、存储引擎层和系统文件层 (1)网络连接层 客户端连接器&#xff08;Client Connectors&#xff09;&#xff1a;提供与MySQL服务器建立的支持。目前几乎支持所有主流的服务端编程技术&#xff0c;例如常…

EE trade:市价建仓的优缺点是什么

在金融市场的复杂环境中&#xff0c;市价建仓策略作为一种常见的交易手段&#xff0c;其优缺点成为了投资者关注的焦点。通过深入分析&#xff0c;我们可以更全面地理解这一策略的利弊&#xff0c;从而在实际操作中做出更加明智的决策。 市价建仓优点分析 快速执行 市价建仓…

鸿蒙系统:未来智能生态的引领者

在当今这个日新月异的互联网领域&#xff0c;操作系统作为连接硬件与软件的桥梁&#xff0c;其重要性不言而喻。随着华为鸿蒙系统&#xff08;HarmonyOS&#xff09;的崛起&#xff0c;一场关于操作系统未来的讨论再次被推向高潮。 鸿蒙OS&#xff0c;华为的全新力作&#xff…

从nginx返回404来看http1.0和http1.1的区别

序言 什么样的人可以称之为有智慧的人呢&#xff1f;如果下一个定义&#xff0c;你会如何来定义&#xff1f; 所谓智慧&#xff0c;就是能区分自己能改变的部分&#xff0c;自己无法改变的部分&#xff0c;努力去做自己能改变的&#xff0c;而不要天天想着那些无法改变的东西&a…

2024年电脑监控软件排行榜(真实测评推荐七款电脑监控软件)

在信息化快速发展的今天&#xff0c;企业对员工电脑活动的监控变得尤为重要。有效的电脑监控软件不仅可以提升员工的工作效率&#xff0c;还能防止信息泄露&#xff0c;保障企业的数据安全。本文将介绍几款知名的电脑监控软件&#xff0c;并对其特点进行详细分析&#xff0c;帮…

JavaWeb系列二十三: web 应用常用功能(文件上传下载)

文章目录 5. 文件上传基本介绍5.1 文件上传-原理示意图5.2 文件上传页面5.3 走通Servlet5.4 表单项区别处理5.5 创建目录-保存文件5.6 中文编码问题5.7 文件上传注意事项和细节5.7.1 按照年月日目录存放5.7.2 文件覆盖问题5.7.3 封装一下 5.8 文件上传其他注意事项5.8.1 upload…

浅谈信息技术高效课堂管理:策略、技巧与实践

引言&#xff1a; 在信息化教育的浪潮中&#xff0c;信息技术课程正逐渐成为学校教育体系中的重要组成部分。然而&#xff0c;信息技术课堂的特殊性——高互动性、高度依赖电子设备&#xff0c;给课堂管理带来了前所未有的挑战。如何在保证教学效率的同时&#xff0c;维护良好…