代理对象与目标对象
1. 定义:代理对象和目标对象
1.1 目标对象(Target Object)
- 目标对象是指 被增强的原始对象,即需要通过 AOP 切面(Aspect)增强功能的业务对象(原始类)。
- 增强逻辑(Advice)最终是应用在目标对象的方法上的。
- 目标对象是实际的业务逻辑持有者,它的功能由我们开发时编写的类直接实现。
例子:
public class UserService {public void createUser() {System.out.println("Creating user...");}
}
UserService是目标对象,定义了业务逻辑。
1.2 代理对象(Proxy Object)
- 代理对象是 Spring AOP 框架生成的 动态代理类实例,它是目标对象的增强版本。
- 代理对象会包装目标对象,并在执行目标对象方法时:
- 增加切面中的增强逻辑(如日志、事务等)。
- 将调用转发到目标对象的方法。
- 代理对象在 Spring 应用上下文中取代了目标对象,用户通过代理对象间接访问目标对象的功能。
代理对象的特点:
- 它的外部表现与目标对象相同(方法签名一致)。
- 它在方法调用时,可以插入增强逻辑(Advice)。
例子:
如果为 UserService 添加一个切面增强功能(例如日志记录),Spring 会生成一个代理对象来拦截方法调用。
UserService proxy = (UserService) context.getBean(UserService.class);
proxy.createUser();
2. 两者的区别
| 特性 | 目标对象(Target Object) | 代理对象(Proxy Object) |
|---|---|---|
| 定义 | 原始的、未增强的业务对象 | Spring AOP 框架生成的动态代理对象 |
| 是否真实存在 | 是真实存在的类实例(用户定义的类) | 是由 Spring 动态生成的新实例 |
| 增强逻辑(Advice) | 不包含增强逻辑 | 包含增强逻辑,方法调用可能被拦截 |
| 访问方式 | 只能通过代理对象间接访问 | 直接被容器返回,外部调用的方法实际上由代理处理 |
| 与 Spring 的关系 | 原始业务逻辑的实现者 | 通过 Spring AOP 动态生成的代理实现 |
| 实际工作 | 执行被调用的具体业务方法 | 拦截方法调用,并决定是否执行增强逻辑或目标对象 |
| 反射行为 | 目标对象的方法是用户定义的,直接反射可访问 | 代理对象的方法是动态生成的,可能有额外逻辑 |
3. Spring 中是如何生成代理对象的
Spring AOP 中的代理对象生成有两种方式,分别基于 JDK 动态代理 和 CGLIB 动态代理。
代理的选择由目标对象的类型决定:
3.1 JDK 动态代理
- 使用 Java 提供的动态代理机制(
java.lang.reflect.Proxy)。 - 要求目标对象实现一个或多个接口。
- 代理对象是目标对象实现的接口的一个实现类。
优点:
- 轻量级,直接基于接口生成代理。
缺点:
- 目标对象必须实现接口,如果是纯类无法使用。
示例:
public interface UserService {void createUser();
}public class UserServiceImpl implements UserService {@Overridepublic void createUser() {System.out.println("Creating user...");}
}
- Spring 会为
UserServiceImpl生成一个动态代理对象,它实现了UserService接口。
3.2 CGLIB 动态代理
- 使用 CGLIB(Code Generation Library)生成目标对象的子类作为代理对象。
- 不要求目标对象实现接口,可以代理普通的类。
- 代理对象是目标对象的子类,并通过方法重写(方法增强)来实现切面功能。
优点:
- 不需要目标对象实现接口,可以直接增强普通类。
缺点:
- 比 JDK 动态代理稍微重一些。
示例:
public class UserService {public void createUser() {System.out.println("Creating user...");}
}
- Spring 会为
UserService生成一个代理子类,并在方法上织入切面逻辑。
3.3 Spring 如何选择代理方式
Spring 默认通过 JDK 动态代理生成代理对象。如果目标对象没有实现接口,则自动切换为 CGLIB 动态代理。
可以通过以下配置强制使用 CGLIB 动态代理:
@EnableAspectJAutoProxy(proxyTargetClass = true)
4. 代理对象和目标对象的实际差异
在 Spring AOP 的运行时动态代理中,外部用户调用的其实是代理对象而非目标对象。以下是一些细节差异:
4.1 方法调用流程
未增强(目标对象直接调用):
UserService userService = new UserService();
userService.createUser();
方法调用的流程:
- 直接调用目标对象的
createUser()方法。 - 输出:
Creating user...
增强(通过代理对象调用):
UserService proxy = (UserService) context.getBean(UserService.class);
proxy.createUser();
方法调用的流程:
- 调用代理对象的
createUser()方法。 - 代理对象拦截方法调用。
- 代理对象决定是否执行切面增强逻辑(如前置通知、后置通知等)。
- 代理对象将调用转发到目标对象的
createUser()方法。
4.2 是否可以直接访问目标对象
-
直接通过 Spring 容器获取的 Bean 是代理对象:
Spring 容器会将代理对象注册为 Bean,用户通过@Autowired或getBean()获取的 Bean 实际上是代理对象。 -
通过 AOP 上下文访问目标对象:
如果需要直接访问目标对象(绕过代理),可以通过 Spring 提供的AopContext:UserService target = (UserService) AopContext.currentProxy();
4.3 代理对象与目标对象的相互关系
-
代理对象包含目标对象:
代理对象会将对业务方法的调用最终转发给目标对象。 -
目标对象不知道代理对象的存在:
目标对象的代码完全独立,不需要感知代理对象或 Spring 的存在。
5. 示例:代理对象和目标对象的工作过程
目标对象
public class UserService {public void createUser() {System.out.println("Creating user...");}
}
切面
@Aspect
@Component
public class LoggingAspect {@Before("execution(* com.example.UserService.*(..))")public void logBefore() {System.out.println("Logging before method execution...");}
}
运行时流程
- Spring 生成一个代理对象(通过 JDK 动态代理或 CGLIB)。
- 用户调用代理对象的
createUser()方法。 - 代理对象拦截该调用,并触发切面逻辑(前置通知)。
- 代理对象将方法调用转发给目标对象。
- 目标对象执行原始业务逻辑。
6. 总结
| 代理对象 | 目标对象 |
|---|---|
| Spring 动态生成的增强版本。 | 开发者定义的原始业务逻辑类。 |
| 包含切面逻辑(通知)。 | 不包含切面逻辑,只有业务逻辑。 |
| 用户通过代理对象间接访问目标对象方法。 | 仅通过代理对象间接调用。 |
| 可以插入增强逻辑(如日志记录、事务)。 | 无法直接应用增强逻辑。 |
相关文章:
代理对象与目标对象
1. 定义:代理对象和目标对象 1.1 目标对象(Target Object) 目标对象是指 被增强的原始对象,即需要通过 AOP 切面(Aspect)增强功能的业务对象(原始类)。增强逻辑(Advice…...
【Kubernetes Pod间通信-第3篇】Kubernetes中Pod与ClusterIP服务之间的通信
引言 我们之前了解了在不同场景下,Kubernetes中Pod之间的通信是如何路由的。 【Kubernetes Pod间通信-第1篇】在单个子网中使用underlay网络实现Pod到Pod的通信【Kubernetes Pod间通信-第2篇】使用BGP实现Pod到Pod的通信现在,我们来看看在集群中,Pod与服务之间的通信是如何…...
DNN(深度神经网络)近似 Lyapunov 函数
import torch import torch.nn as nn import torch.optim as optim import matplotlib.pyplot as plt # from torchviz import make_dot import torchviz# 1. Lyapunov 函数近似器(MLP 结构) class LyapunovNet(nn.Module):def __init__(self, input_dim…...
128陷阱
首先我们了解一下关于包装器类型 java是面向对象的语言,但基本类型并不是面向对象的,从而出现了包装器类型,并且包装器添加了更多的属性和方法。如我们在使用集合类型Collection的时候就一定要使用包装类型而非基本类型,它相当于将…...
PromptSource和LangChain哪个更好
目录 1. 设计目标与定位 PromptSource LangChain 2. 功能对比 3. 优缺点分析 PromptSource LangChain 4. 如何选择? 5. 总结 PromptSource 和 LangChain 是两个在自然语言处理(NLP)领域非常有用的工具,但它们的设计目标和…...
构成正方形的数量:算法深度剖析与实践
目录 引言算法核心概念 定义正方形的构成条件数据结构与输入形式算法数学原理 几何关系的数学表达坐标运算与判定逻辑Python 实现 代码展示代码解析Python 实现的优势与局限C 语言实现 代码展示代码解析C 语言实现的性能特点性能分析与优化 性能分析 时间复杂度空间复杂度优化思…...
Redis持久化-秒杀系统设计
在构建高性能、高可用的系统时,Redis 作为缓存和消息队列的角色越来越重要。在一些场景下,我们还需要将 Redis 的数据进行持久化,以确保数据的安全性和恢复能力。除此之外,秒杀系统也越来越成为电商、抢购等平台的核心功能之一。本…...
音视频入门基础:RTP专题(8)——使用Wireshark分析RTP
一、引言 通过Wireshark可以抓取RTP数据包,该软件可以从Wireshark Go Deep 下载。 二、通过Wireshark抓取RTP数据包 首先通过FFmpeg将一个媒体文件转推RTP,生成RTP流: ffmpeg -re -stream_loop -1 -i input.mp4 -vcodec copy -an -f rtp …...
OpenAI 实战进阶教程 - 第六节: OpenAI 与爬虫集成实现任务自动化
爬虫与 OpenAI 模型结合,不仅能高效地抓取并分析海量数据,还能通过 NLP 技术生成洞察、摘要,极大提高业务效率。以下是一些实际工作中具有较高价值的应用案例: 1. 电商价格监控与智能分析 应用场景: 电商企业需要监控…...
SpringUI Web高端动态交互元件库
Axure Web高端动态交互元件库是一个专为Web设计与开发领域设计的高质量资源集合,旨在加速原型设计和开发流程。以下是关于这个元件库的详细介绍: 一、概述 Axure Web高端动态交互元件库是一个集成了多种预制、高质量交互组件的工具集合。这些组件经过精…...
解密企业安全密码:密钥管理服务如何重塑数据保护?
在数字化时代,数据是企业最宝贵的资产之一。然而,随着网络威胁的不断升级和数据泄露事件的频繁发生,如何保护企业数据的安全已成为每个组织面临的紧迫问题。传统的安全措施往往无法应对复杂的威胁环境,密钥管理服务作为企业信息安…...
基于keepalived+GTID半同步主从复制的高可用MySQL集群
文章目录 项目架构图项目名称项目环境项目描述ip地址规划项目步骤一.安装好8台全新的centos7.9的系统,关闭firewalld和selinux,配置每台主机的静态ip地址,设置每台主机对应的主机名。1、关闭firewalld2.关闭seLinux3.配置每台主机静态ip地址4…...
图片PDF区域信息批量提取至Excel,基于QT和阿里云api的实现方案
办公文档处理:在企业日常办公中,经常会遇到大量的扫描文档(如发票、合同、报表等)以图片或 PDF 格式存储。需要将这些文档中的特定区域信息(如发票金额、合同条款、报表数据等)提取出来,整理到 …...
Java 大视界 -- Java 大数据在智能教育中的应用与个性化学习(75)
💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也期待你毫无保留地分享独特见解,愿我们于此携手成长,共赴新程!💖 一、…...
从零手写Spring IoC容器(二):bean的定义与注册
从零手写Spring IoC容器(二):bean的定义与注册 一. 回顾简单容器的不足之处 在第一章中,我们实现了一个最简单的 IoC 容器,但该版本存在诸多不足,例如: Bean 的管理方式过于简单,…...
《大模型面试宝典》(2025版) 发布了
基于去年我们写的《大模型面试宝典》(2024版)的基础上,我根据自己实践经验和星球小伙伴的面经分享总结推出《大模型面试宝典》(2025版),共计52w字。 与去年相比,内容增加了星球成员面试真题分享、大模型最新考试要点总结、DeepSeek 项目实战…...
AWS门店人流量数据分析项目的设计与实现
这是一个AWS的数据分析项目,关于快消公司门店手机各个门店进店人流量和各个产品柜台前逗留时间(利用IoT设备采集)和销售数据之间的统计分析,必须用到但不限于Amazon Kensis Data Stream,Spark Streaming,Sp…...
出租车特殊计费表算法解析与实现
目录 引言算法核心概念 特殊计费规则解析数据类型与输入输出算法数学原理 数字位判断与处理逻辑数值转换与累加计算算法框架图Python 实现 代码展示代码解析Python 实现的优势与局限C 语言实现 代码展示代码解析C 语言实现的性能特点性能分析与优化 性能分析 时间复杂度空间复杂…...
文档解析技术:如何高效提取PDF扫描件中的文字与表格信息?
想要高效提取PDF扫描件中的文字与表格信息,通常需要借助专业的工具或在线服务,以下是一些可行的方法: 预处理扫描件:在提取文字之前,尽量确保扫描件的图像质量清晰。如果扫描件模糊或有污渍,可以使用图像处…...
【2】高并发导出场景下,服务器性能瓶颈优化方案-异步导出
Java 异步导出是一种在处理大量数据或复杂任务时优化性能和用户体验的重要技术。 1. 异步导出的优势 异步导出是指将导出操作从主线程中分离出来,通过后台线程或异步任务完成数据处理和文件生成。这种方式可以显著减少用户等待时间,避免系统阻塞&#x…...
云原生核心技术 (7/12): K8s 核心概念白话解读(上):Pod 和 Deployment 究竟是什么?
大家好,欢迎来到《云原生核心技术》系列的第七篇! 在上一篇,我们成功地使用 Minikube 或 kind 在自己的电脑上搭建起了一个迷你但功能完备的 Kubernetes 集群。现在,我们就像一个拥有了一块崭新数字土地的农场主,是时…...
【Linux】shell脚本忽略错误继续执行
在 shell 脚本中,可以使用 set -e 命令来设置脚本在遇到错误时退出执行。如果你希望脚本忽略错误并继续执行,可以在脚本开头添加 set e 命令来取消该设置。 举例1 #!/bin/bash# 取消 set -e 的设置 set e# 执行命令,并忽略错误 rm somefile…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
LeetCode - 394. 字符串解码
题目 394. 字符串解码 - 力扣(LeetCode) 思路 使用两个栈:一个存储重复次数,一个存储字符串 遍历输入字符串: 数字处理:遇到数字时,累积计算重复次数左括号处理:保存当前状态&a…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个生活电费的缴纳和查询小程序
一、项目初始化与配置 1. 创建项目 ohpm init harmony/utility-payment-app 2. 配置权限 // module.json5 {"requestPermissions": [{"name": "ohos.permission.INTERNET"},{"name": "ohos.permission.GET_NETWORK_INFO"…...
在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...
第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词
Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵,其中每行,每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid,其中有多少个 3 3 的 “幻方” 子矩阵&am…...
