SpringAOP前置——代理模式

server/2025/1/17 14:24:16/

代理模式是SpringAOP(面向切面编程)的底层原理

代理模式的分类

  • 静态代理
  • 动态代理

静态代理

角色分析:
抽象角色:一般使用抽象类或接口来解决
代理角色:代理真实角色,在代理真实角色后,一般会做一些附属操作
真实角色:被代理的角色
客户:访问代理对象的角色,可以理解为一个处理事务的线程,多为一次业务处理

以租房举例子进行理解

房东有房子要出租,将房源信息告诉中介,也就是让中介代理房东进行房屋租赁这件事。房东是真实角色(被代理的角色),中介是代理角色,房屋租赁这件事是抽象角色。
租房者是客户,相当于一次租房业务。
接下来上代码
抽象角色:房屋租赁这件事

java">package com.zbt.proxy.demo01;/*** @author* @createAt 2025/1/12 20:58*//*** 租房 接口*/
public interface Rent {//出租事件 出租的是房子 对应着一个房东void rent(Host host);
}

真实角色:房东 有两个属性 一个名字,一个房屋地址

java">package com.zbt.proxy.demo01;/*** @author* @createAt 2025/1/12 20:59*//*** 房东*/
public class Host implements Rent{private String name;private String address;public Host() {}public Host(String name, String address) {this.name = name;this.address = address;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getAddress() {return address;}public void setAddress(String address) {this.address = address;}public void rent(Host host) {System.out.println("我是房东:"+name+",我要出租房子,地址是:"+address);}
}

代理角色:中介

java">package com.zbt.proxy.demo01;/*** @author* @createAt 2025/1/12 21:01*/import java.util.ArrayList;
import java.util.List;/*** 中介 代理租房*/
public class Proxy implements Rent{//被代理的对象 对于中介来说就是一个房东private Host;public void setHost(Host host){hostLs.add(host);}public Host getHost(){return host;}public void rent(Host host) {//中介知道房东要出租房子  也就是中介对于你来说 代理了这套房子的房东host.rent(host);//带你去看房子seeHouse(host);//你看中了房子 签合同hetong();//中介收了中介服务费fare();}public void seeHouse(Host host){System.out.println("中介带你看了在"+host.getAddress()+"的房子");}public void hetong(){System.out.println("签租赁合同");}public void fare(){System.out.println("收中介费");}}

客户:你 你要租房子 触发了一次租房业务流程(main方法)

java">package com.zbt.proxy.demo01;/*** @author* @createAt 2025/1/12 21:00*//*** 租客要租房*/
public class Client {public static void main(String[] args) {//被代理对象 -- 房东Host host1 = new Host("房东1号","南京南街5-5-1号");//代理对象 -- 中介Proxy proxy;= new Proxy();//房东在中介登记房子proxy.setHostLs(host3);//你要租房子  找到了中介 触发了租房流程//中介给你介绍了一套房子Host host = proxy.getHostLs(0);//中介带你走租房流程  看房 签合同proxy.rent(host);}
}

静态代理模式的好处

  • 可以使真实角色的“动作”更加纯粹:例如上面的例子,房东就只需要找中介登记一下房源,就可以等着房子出租出去了,不需要进行公共业务(带人看房,贴租房广告等)
  • 公共业务交给代理角色来进行,进行过了业务的分工,
  • 在公共业务需要进行扩展的时候方便集中管理,只需要改代理对象就可以了。
    静态代理模式缺点
    一个真实角色就会产生一个代理角色,代码量会增加,开发效率降低

模拟实际开发逻辑理解静态代理

实际开发经常用到用户信息的增删查改
我们通常的写法是:

java">package com.zbt.proxy.demo02;/*** 接口类*/
public interface UserService {void insert();void delete();void update();void select();
}package com.zbt.proxy.demo02;/*** @author* @createAt 2025/1/12 22:01*//*** 实现类*/
public class UserServiceImpl implements UserService{public void insert() {System.out.println("插入了一个用户");}public void delete() {System.out.println("删除了一个用户");}public void update() {System.out.println("修改了一个用户");}public void select() {System.out.println("查询了一个用户");}
}package com.zbt.proxy.demo02;/*** 调用* @author* @createAt 2025/1/12 22:02*/public class Client {public static void main(String[] args) {UserServiceImpl userService = new UserServiceImpl();userService.insert();}
}

这样当然可以实现业务需求,但是当后期,如果在此基础上需要增加新的需求时,就得修改原有的业务逻辑代码,例如在调用每个方法时,增加一条日志记录,我们就需要去修改原来的UserServiceImpl 里面每一个方法,添加一条打印语句。如果需要进行的改较为复杂,那么修改原业务逻辑是致命的,此时我们将写法改为静态代理模式
在原有基础上,增加一个代理类UserServiceProxy 代理UserServiceImpl

java">package com.zbt.proxy.demo02;/*** 代理类  代理UserServiceImpl  将UserServiceImpl 对象通过set方法传入 里面直接通过UserServiceImpl 对象调用对应方法,需要修改逻辑时,直接修改代理类,例如上面提到的增加一条调用日志打印* @author* @createAt 2025/1/12 22:03*/public class UserServiceProxy implements UserService{private UserServiceImpl userService;public UserServiceImpl getUserService() {return userService;}public void setUserService(UserServiceImpl userService) {this.userService = userService;}public void insert() {log("insert");userService.insert();}public void delete() {log("delete");userService.delete();}public void update() {log("update");userService.update();}public void select() {log("select");userService.select();}//日志打印方法public void log(String msg){System.out.println("调用了"+msg+"方法");}
}package com.zbt.proxy.demo02;/*** 调用* @author* @createAt 2025/1/12 22:02*/public class Client {public static void main(String[] args) {UserServiceImpl userService = new UserServiceImpl();UserServiceProxy proxy = new UserServiceProxy();proxy.setUserService(userService);proxy.insert();}
}

动态代理

  • 动态代理和静态代理角色分类一样
  • 动态代理的代理类是动态生成的,不是提前写好的
  • 动态代理分为两大类:基于接口的动态代理(JDK的动态代理)、基于类的动态代理(cglib)
    两个关键的类:Proxy、InvocationHandler
  • Proxy用来生成动态代理实例
  • InvocationHandler用来调用处理程序并返回结果
    上代码:还是上面用户信息的增删查改的例子
java">package com.zbt.proxy.demo03;/*** 接口类*/
public interface UserService {void insert();void delete();void update();void select();
}
package com.zbt.proxy.demo03;/*** @author* @createAt 2025/1/12 22:01*//*** 实现类*/
public class UserServiceImpl implements UserService {public void insert() {System.out.println("插入了一个用户");}public void delete() {System.out.println("删除了一个用户");}public void update() {System.out.println("修改了一个用户");}public void select() {System.out.println("查询了一个用户");}
}
package com.zbt.proxy.demo03;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;/*** 动态获得代理类的工具类* @author* @createAt 2025/1/13 21:04*/
public class ProxyInvocationHandler implements InvocationHandler {//被代理的接口——真实对象private Object target;public void setTarget(Object target) {this.target = target;}//生成得到代理类实例public Object getProxy(){return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(), this);}//处理代理实例 并返回结果public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {log(method.getName());//method.getName() 获取调用的方法名return method.invoke(target, args);}public void log(String msg){System.out.println("调用了"+msg+"方法");}
}
package com.zbt.proxy.demo03;import com.zbt.proxy.demo02.UserServiceProxy;/*** @author* @createAt 2025/1/12 22:02*/public class Client {public static void main(String[] args) {//真实角色UserServiceImpl userService = new UserServiceImpl();//代理角色 原本并不存在的类,动态生成的类ProxyInvocationHandler handler = new ProxyInvocationHandler();handler.setTarget(userService);//生成动态代理类UserService proxy = (UserService)handler.getProxy();//调用方法proxy.insert();}
}

这个动态代理我也没太理解。。。。等我再研究研究,应该就是利用反射的机制,借用Proxy、InvocationHandler来动态生成代理类


http://www.ppmy.cn/server/159105.html

相关文章

【深度学习】关键技术-正则化(Regularization)

正则化(Regularization) 是一种用于防止模型过拟合的技术。它通过在损失函数中添加额外的约束项,限制模型的复杂度,从而提高模型的泛化能力。 正则化的主要作用 防止过拟合:通过抑制模型对训练数据的过度拟合&#xf…

【遥感目标检测】【数据集】DOTA:用于航空图像中目标检测的大规模数据集

DOTA:A large-scale dataset for object detection in aerial images DOTA:用于航空图像中目标检测的大规模数据集 CVPR 2018 论文地址 数据集地址 0.论文摘要 目标检测是计算机视觉中一个重要而富有挑战性的问题。尽管过去十年见证了自然场景中目标检测的重大进展…

aws(学习笔记第二十四课) 使用sam开发step functions

aws(学习笔记第二十四课) 使用sam开发step functions 学习内容: 生成sam的step functions实例程序什么是SAM amazon Serverless Application ModelSAM程序结构SAM执行程序 1. 生成sam的step functions实例程序 参照文档 这里参照AWS的官方文档SAM amazon Serverl…

Windows图形界面(GUI)-QT-C/C++ - Qt控件与布局系统详解

公开视频 -> 链接点击跳转公开课程博客首页 -> ​​​链接点击跳转博客主页 目录 Qt布局系统(Layouts) 布局管理器基础 高级布局技巧 嵌套布局 设置间距和边距 常用控件详解 按钮类控件 QPushButton (标准按钮) QRadioButton (单选按钮) QCheckBox (复选框) …

GaussDB中的Vacuum和Analyze

GaussDB中的Vacuum和Analyze 基本概念与区别手动Vacuum和Analyze查看Vacuum和Analyze记录Autovacuum配置参数 基本概念与区别 使用VACUUM、VACUUM FULL和ANALYZE命令定期对每个表进行维护,主要有以下原因: VACUUM FULL可回收已更新或已删除的数据所占据…

日拱一卒(20)——leetcode学习记录:大小为 K 且平均值大于等于阈值的子数组数目

一、题目 给定数组,统计数组中长度为k的子数组且该子数组的平均值大于threshold的数量 二、思路 滑动窗思路,计算长度为k的滑动窗的平均值,关键点在于,每滑动一次,只需要去掉头增加尾,而不需要重新全部计…

vue运用uniapp框架开发企业微信小程序中常用的一些基础方法

嗨,我是小路。今天主要和大家分享的主题是“vue运用uniapp框架开发企业微信小程序中常用的一些基础方法”。 作为一名程序员,很多代码都是忘了再用,用了再忘。 今天梳理下日常开发中常用到的一些基础的方法,以方便后期开…

2Hive表类型

2Hive表类型 1 Hive 数据类型2 Hive 内部表3 Hive 外部表4 Hive 分区表5 Hive 分桶表6 Hive 视图 1 Hive 数据类型 Hive的基本数据类型有:TINYINT,SAMLLINT,INT,BIGINT,BOOLEAN,FLOAT,DOUBLE&a…