当前位置: 首页 > news >正文

深入理解静态代理与JDK动态代理

代理模式

简介

代理模式是一种常用的设计模式,在AOP、RPC等诸多框架中均有它的身影

  • 代理对象存在的价值主要用于拦截对真实业务对象的访问;
  • 代理对象具有和目标对象(真实业务对象)实现共同的接口或继承于同一个类;
  • 代理对象是对目标对象的增强,以便对消息进行预处理和后处理。

定义与结构

定义 为其他对象提供一种代理以控制对这个对象的访问。

代理模式主要包含三个角色,即抽象主题角色(Subject)、委托类角色(被代理角色,Proxied)以及代理类角色(Proxy),如上图所示

  • 抽象主题角色:可以是接口,也可以是抽象类;
  • 委托类角色:真实主题角色,业务逻辑的具体执行者;
  • 代理类角色:内部含有对真实对象RealSubject的引用,负责对真实主题角色的调用,并在真实主题角色处理前后做预处理和后处理。

静态代理实现

静态代理三步走

1. 定义业务接口

public interface HelloService {
 String hello(String name);
 String hi(String msg);
}
复制代码

2.实现业务接口

public class HelloServiceImpl implements HelloService{
 @Override
 public String hello(String name) {
 return "Hello " + name;
 }
 @Override
 public String hi(String msg) {
 return "Hi, " + msg;
 }
}
复制代码

3.理类并实现业务接口

public class HelloServiceProxy implements HelloService {
 private HelloService helloService;
 public HelloServiceProxy(HelloService helloService) {
 this.helloService = helloService;
 }
 @Override
 public String hello(String name) {
 System.out.println("预处理...");
 String result = helloService.hello(name);
 System.out.println(result);
 System.out.println("后处理...");
 return result;
 }
 @Override
 public String hi(String msg) {
 System.out.println("预处理...");
 String result = helloService.hi(msg);
 System.out.println(result);
 System.out.println("后处理...");
 return result;
 }
}
复制代码

最后便可通过客户端进行调用

public class Main {
 public static void main(String[] args){
 HelloService helloService = new HelloServiceImpl();
 HelloServiceProxy helloServiceProxy = new HelloServiceProxy(helloService);
 helloServiceProxy.hello("Panda");
 helloServiceProxy.hi("Panda");
 }
}
复制代码

JDK 动态代理

动态代理可以很方便地对委托类的相关方法进行统一增强处理,如添加方法调用次数、添加日志功能等等。动态代理主要分为JDK动态代理和cglib动态代理两大类,本文主要对JDK动态代理进行探讨

JDK动态代理使用步骤

  • 创建被代理的接口和类
// 抽象主题角色
public interface HelloService {
 String hello(String name);
 String hi(String msg);
}
// 具体(真实)主题角色
public class HelloServiceImpl implements HelloService{
 @Override
 public String hello(String name) {
 return "Hello " + name;
 }
 @Override
 public String hi(String msg) {
 return "Hi, " + msg;
 }
}
复制代码
  • 实现InvocationHandler接口
public class MyInvocationHandler implements InvocationHandler{
 // 真实业务对象
 private Object target;
 public MyInvocationHandler(Object target){
 this.target = target;
 }

 @Override
 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
 // 增强逻辑
 System.out.println("PROXY : " + proxy.getClass().getName());
 // 反射调用,目标方法
 Object result = method.invoke(target, args);
 // 增强逻辑
 System.out.println(method.getName() + " : " + result);
 return result;
 }
}
复制代码
  • 创建代理类并生成相应的代理对象
// 生成代理类的class对象
Class<?> clazz = Proxy.getProxyClass(helloService.getClass().getClassLoader(), helloService
 .getClass().getInterfaces());
// 创建InvocationHandler
InvocationHandler myInvocationHandler = new MyInvocationHandler(helloService);
// 获取代理类的构造器对象
Constructor constructor = clazz.getConstructor(new Class[] {InvocationHandler.class});
// 反射创建代理对象
HelloService proxy = (HelloService)constructor.newInstance(myInvocationHandler);
复制代码
  • 使用代理
proxy.hello("rico");
proxy.hi("panda");
复制代码

个人看点:

  1. 实现动态代理的关键技术是反射;
  2. 代理对象是对目标对象的增强,以便对消息进行预处理和后处理;
  3. InvocationHandler中的invoke()方法是代理类完整逻辑的集中体现,包括要切入的增强逻辑和进行反射执行的真实业务逻辑;
  4. 使用JDK动态代理机制为某一真实业务对象生成代理,只需要指定目标接口、目标接口的类加载器以及具体的InvocationHandler即可。
  5. JDK动态代理的典型应用包括但不仅限于AOP、RPC、Struts2、Spring等重要经典框架。

转载于:https://juejin.im/post/5caf028ae51d456e2c2484e3

相关文章:

  • 常见漏洞解析
  • [转] 谈谈前端异常捕获与上报
  • Apache Tomcat 8.5.40 与 7.0.94 发布
  • 不要一棍子打翻所有黑盒模型,其实可以让它们发挥作用 ...
  • 线程同步利与弊,线程同步的前提
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • Python数据类型、运算符、语句、循环
  • 激活效能,CODING 敏捷研发模块上线
  • cmd中subst的使用
  • [MySQL光速入门]003 留点作业...
  • C# - 为值类型重定义相等性
  • Es6初级入门(一)
  • thinkphp+redis实现秒杀,缓存等功能
  • (JS基础)String 类型
  • django2中表单的使用二
  • 《Javascript数据结构和算法》笔记-「字典和散列表」
  • 〔开发系列〕一次关于小程序开发的深度总结
  • ECMAScript 6 学习之路 ( 四 ) String 字符串扩展
  • es6要点
  • Go 语言编译器的 //go: 详解
  • Iterator 和 for...of 循环
  • SpingCloudBus整合RabbitMQ
  • vue和cordova项目整合打包,并实现vue调用android的相机的demo
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • ------- 计算机网络基础
  • 马上搞懂 GeoJSON
  • 它承受着该等级不该有的简单, leetcode 564 寻找最近的回文数
  • mysql 慢查询分析工具:pt-query-digest 在mac 上的安装使用 ...
  • postgresql行列转换函数
  • ​总结MySQL 的一些知识点:MySQL 选择数据库​
  • #14vue3生成表单并跳转到外部地址的方式
  • #Js篇:单线程模式同步任务异步任务任务队列事件循环setTimeout() setInterval()
  • (14)Hive调优——合并小文件
  • (2009.11版)《网络管理员考试 考前冲刺预测卷及考点解析》复习重点
  • (7)STL算法之交换赋值
  • (ISPRS,2023)深度语义-视觉对齐用于zero-shot遥感图像场景分类
  • (原創) 人會胖會瘦,都是自我要求的結果 (日記)
  • (转)eclipse内存溢出设置 -Xms212m -Xmx804m -XX:PermSize=250M -XX:MaxPermSize=356m
  • (转)使用VMware vSphere标准交换机设置网络连接
  • .bat批处理出现中文乱码的情况
  • .NET Conf 2023 回顾 – 庆祝社区、创新和 .NET 8 的发布
  • .net core 依赖注入的基本用发
  • .NET Micro Framework初体验(二)
  • .NET/C# 使用 #if 和 Conditional 特性来按条件编译代码的不同原理和适用场景
  • .net和jar包windows服务部署
  • @cacheable 是否缓存成功_Spring Cache缓存注解
  • @软考考生,这份软考高分攻略你须知道
  • [ActionScript][AS3]小小笔记
  • [android] 请求码和结果码的作用
  • [AX]AX2012开发新特性-禁止表或者表字段
  • [C# 网络编程系列]专题六:UDP编程
  • [CentOs7]iptables防火墙安装与设置
  • [corCTF 2022] CoRJail: From Null Byte Overflow To Docker Escape
  • [docker]docker网络-直接路由模式
  • [ffmpeg] 定制滤波器