设计模式之【动态代理】

目录

动态代理中存在的概念

JDK动态代理

代理工厂【ProxyFactory】实现【InvocationHandler】

目标类的接口【TargetInterface】

目标类【Target】实现了接口

测试类【JDKDynamicProxyTest】

CGLIB动态代理

添加Maven依赖

代理工厂【ProxyFactory】实现【MethodInterceptor】

目标类【Target】没有实现接口 

 测试类【CGLIBDynamicProxyTest】


动态代理中存在的概念

  • 目标类、目标对象【target】
  • 代理类、代理对象【proxy】
  • 增强逻辑【advice】

JDK动态代理

  • 要求:目标类必须要实现接口
  • 原理:和目标类实现同一个接口,对目标类进行增强

代理工厂【ProxyFactory】实现【InvocationHandler】

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;public class ProxyFactory<T> implements InvocationHandler {// 目标对象private final T target;public ProxyFactory(T target) {this.target = target;}/*** 获取代理对象*/@SuppressWarnings("all")public T getProxy() {// 类加载器ClassLoader classLoader = getClassLoader();// 接口Class<?>[] interfaces = getInterfaces(target);// 返回代理对象return (T) Proxy.newProxyInstance(classLoader, interfaces, this);}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {Object obj;try {// 前置增强before();obj = method.invoke(target, args);// 返回后增强afterReturning();return obj;} catch (Throwable throwable) {// 异常后增强afterThrowing();throw throwable;} finally {// 后置增强after();}}private ClassLoader getClassLoader() {return Thread.currentThread().getContextClassLoader();}private Class<?>[] getInterfaces(T target) {return target.getClass().getInterfaces();}private void before() {System.out.println("before...");}private void after() {System.out.println("after...");}private void afterThrowing() {System.out.println("afterThrowing...");}private void afterReturning() {System.out.println("afterReturning...");}
}

目标类的接口【TargetInterface】

public interface TargetInterface {void method();
}

目标类【Target】实现了接口

public class Target implements TargetInterface {public void method() {System.out.println("执行目标方法...");}
}

测试类【JDKDynamicProxyTest】

/*** 测试JDK动态代理*/
public class JDKDynamicProxyTest {public static void main(String[] args) {// 目标对象TargetInterface target = new Target();ProxyFactory<TargetInterface> proxyFactory = new ProxyFactory<>(target);// 代理对象TargetInterface proxy = proxyFactory.getProxy();// 目标对象方法执行逻辑 + 增强逻辑proxy.method();}
}

CGLIB动态代理

  • 要求:目标类非final关键字修饰
  • 原理:将目标类作为父类,通过子类来对父类进行增强

添加Maven依赖

<!--CGLIB依赖-->
<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>3.3.0</version>
</dependency>

代理工厂【ProxyFactory】实现【MethodInterceptor】

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;public class ProxyFactory<T> implements MethodInterceptor {/*** 获取代理对象*/@SuppressWarnings("all")public T getProxy(Class<T> cls) {Enhancer enhancer = new Enhancer();// 设置父类enhancer.setSuperclass(cls);// 设置回调enhancer.setCallback(this);// 返回创建的代理对象return (T) enhancer.create();}@Overridepublic Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {Object result;try {System.out.println("前置增强...");// 调用目标对象的方法result = methodProxy.invokeSuper(o, args);System.out.println("正常返回时增强...");return result;} catch (Throwable throwable) {System.out.println("异常时增强...");throw throwable;} finally {System.out.println("后置增强...");}}
}

目标类【Target】没有实现接口 

public class Target {public void method() {System.out.println("目标方法执行++++++++");}
}

 测试类【CGLIBDynamicProxyTest】

public class CGLIBDynamicProxyTest {public static void main(String[] args) {ProxyFactory<Target> proxyFactory = new ProxyFactory<>();// 代理对象Target proxy = proxyFactory.getProxy(Target.class);// 原始逻辑 + 增强逻辑proxy.method();}
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Linux驱动-快速回顾】一次性快速回顾TTY体系知识点(新手友好)

我将遵循一条严格的“问题驱动”和“演进”的逻辑线索来构建整个TTY知识体系。每引入一个新概念&#xff0c;都是为了解决前一个阶段出现的问题。这样&#xff0c;你不仅能知道“是什么”&#xff0c;更能深刻理解“为什么是这样设计的”。 第〇阶段&#xff1a;最原始的需求 …

深入浅出:让机器听懂世界的耳朵——梅尔频率倒谱系数(MFCCs)

深入浅出&#xff1a;让机器听懂世界的耳朵——梅尔频率倒谱系数&#xff08;MFCCs&#xff09; 在人工智能的浪潮中&#xff0c;语音识别、声纹支付、音乐推荐等技术早已融入我们的日常生活。你是否曾好奇&#xff0c;计算机是如何理解并区分各种复杂的声音信号的&#xff1f;…

Ubuntu22.04安装/使用Gazebo时踩的一些坑

首先&#xff0c;本人原本打算安装gazebo11的&#xff0c;因为官方好像不支持ubuntu22.04&#xff0c;所以要通过PPA和ROS2 humble来安装&#xff0c;安装过程跟着教程来的&#xff0c;也就是下面这篇 ubuntu22.04安装gazebo11&#xff08;ROS2 Humble&#xff09;-CSDN博客 …

CPT203-Software Engineering: Introduction 介绍

目录 1.专业名词定义 1.1计算机软件的定义 1.2软件系统的定义 1.3软件工程的定义 2.软件的失败与成功 2.1 失败 2.2 成功 3.软件开发 Professional software development 3.1 分类 3.2 专业软件开发 professional software development 3.3专业软件开发产品特性 3.4…

诊断工程师进阶篇 --- 车载诊断怎么与时俱进?

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 做到欲望极简,了解自己的真实欲望,不受外在潮流的影响,不盲从,不跟风。把自己的精力全部用在自己。一是去掉多余,凡事找规律,基础是诚信;二是…

奥特曼论人工智能、OpenAI与创业

来自Y Combinator的YouTube视频&#xff0c;展示了OpenAI首席执行官萨姆奥特曼分享的深刻见解。他讨论了OpenAI从一个看似疯狂的通用人工智能&#xff08;AGI&#xff09;梦想&#xff0c;如何发展成为一个全球性的现象。奥特曼强调了早期决策的关键性、吸引顶尖人才的策略&…

React Ref使用

受控与非受控组件 Ref 1.获取原生dom 类组件中&#xff1a;在componentDidMount方法内使用document.getElementById的方法获取到dom元素 1 目标dom增加ref属性 设置为字符串 <h2 reftitleref></h2>function changeRef(){this.refs.titleref.innerHtml }2 函数组件…

地下管线安全的智能监测先锋:智能标志桩图像监测装置解析​

​在城市与乡村的地下&#xff0c;纵横交错的管线是能源与信息传输的关键通道。但深埋地下的电缆、燃气管道等设施&#xff0c;因难以直观监测&#xff0c;面临施工误挖、自然灾害等风险。传统防护手段力不从心&#xff0c;TLKS-PAZ01 智能标志桩图像监测装置的诞生&#xff0c…

Camera相机人脸识别系列专题分析之十六:人脸特征检测FFD算法之libcvface_api.so数据结构详细注释解析

【关注我&#xff0c;后续持续新增专题博文&#xff0c;谢谢&#xff01;&#xff01;&#xff01;】 上一篇我们讲了&#xff1a; 这一篇我们开始讲&#xff1a; Camera相机人脸识别系列专题分析之十六&#xff1a;人脸特征检测FFD算法之libcvface_api.so数据结构详细注释解析…

【字节跳动】数据挖掘面试题0012:数据分析、数据挖掘、数据建模的区别

文章大纲 数据分析、数据挖掘、数据建模的区别一、核心定义与目标二、技术方法差异三、应用场景对比四、三者的关联与递进关系五、面试应答策略 数据分析、数据挖掘、数据建模的区别 一、核心定义与目标 数据分析&#xff1a; 是对已有的数据进行收集、清洗、整理&#xff0c;并…

预警:病毒 “黑吃黑”,GitHub 开源远控项目暗藏后门

在开源生态蓬勃发展的当下&#xff0c;黑客们也将黑手伸向了代码共享平台。当黑产开发者以为在共享 “行业秘笈” 时&#xff0c;殊不知已经掉入了黑客布置的陷阱 —— 看似方便的后门远程控制源码和游戏作弊外挂源码等 “圈内资源”&#xff0c;实则是植入了恶意代码的投毒诱饵…

Qt中的QProcess类

Qt中的QProcess类 QProcess 是 Qt 框架中用于启动和控制外部进程的类&#xff0c;它属于 QtCore 模块。这个类提供了执行外部程序并与它们交互的功能。 一、主要功能 启动外部程序&#xff1a;可以启动系统上的其他可执行程序进程通信&#xff1a;通过标准输入、输出和错误流…

周任务自动化升级:N8N与多维表格无缝联动全解析

.自动化之言&#xff1a; 在上一篇文章中&#xff0c;我们介绍了如何利用多维表格&#xff08;如飞书多维表格或Notion&#xff09;搭建一个灵活的任务管理系统。现在我们将进一步扩展这个系统&#xff0c;借助 N8N 实现周报的自动汇总与邮件发送&#xff0c;真正实现任务管理…

Go语言的web框架--gin

本章内容&#xff0c;会介绍一下gin的运用&#xff0c;以及gin框架底层的内容&#xff0c;话不多说&#xff0c;开始进入今天的主题吧&#xff01; 一.基本使用 gin框架支持前后端不分离的形式&#xff0c;也就是直接使用模板的形式。 模板是什么&#xff1f; 这里可能有同…

企业为什么需要双因素认证?

从进入互联网时代开始&#xff0c;密码是我们个人日常的重要保护。但是单独的密码保护可能已经不再适应当前的数字化时代。密码已经不再足够安全最近发生的各种安全漏洞让我重新审视网络安全。几行代码可能就导致了全球数以百万的登录凭证被泄露。今天&#xff0c;仅仅周期性地…

Spring Boot + 本地部署大模型实现:优化与性能提升!

在Spring Boot中集成本地部署的大模型&#xff08;如LLaMA、ChatGLM等&#xff09;并进行优化&#xff0c;需要从模型选择、推理加速、资源管理和架构设计等多方面入手。以下是完整的优化方案及实现步骤&#xff1a; 一、核心优化策略 1. 模型量化 目标&#xff1a;减少显存占…

仿mudou库one thread oneloop式并发服务器

前言 我们所要实现的是一个高并发服务器的组件&#xff0c;使服务器的性能更加高效&#xff0c;是一个高并发服务器的组件&#xff0c;并不包含实际的业务。 首先需要先明确我们所要实现的目标是什么 第一点&#xff0c;实现一个高并发的服务器第二点&#xff0c;在服务器的基础…

超详细的私有化安装部署Dify服务以及安装过程中问题处理

一、什么是Dify Dify 是一款开源的大语言模型(LLM) 应用开发平台。它融合了后端即服务&#xff08;Backend as Service&#xff09;和 LLMOps 的理念&#xff0c;使开发者可以快速搭建生产级的生成式 AI 应用。即使你是非技术人员&#xff0c;也能参与到 AI 应用的定义和数据…

国产DSP,QXS320F280049,QXS320F28377D,QXS320F2800137,QXS320F28034

自定义指令集&#xff0c;自研内核架构&#xff0c;基于eclipse自研IDE&#xff0c;工具链&#xff0c;算法库。 根据自研QXS320F280049&#xff0c;做了600W和2KW数字电源方案&#xff0c;1.5KW电机方案&#xff0c;目前已在市场大量投产。 QXS320F290049应用于数字电源&#…

dotnet publish 发布后的项目,例如asp.net core mvc项目如何在ubuntu中运行,并可外部访问

复制到 Ubuntu 上的是使用 Visual Studio 或 dotnet publish 命令生成的 发布后的输出文件&#xff08;publish output&#xff09;&#xff0c;而不是原始项目源代码。在这种情况下&#xff0c;确实没有 .csproj 文件&#xff0c;所以不能直接用 dotnet run 启动。但你可以通过…