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

Java 动态代理教程(JDK 动态代理)(以RPC 过程为例)

1. 什么是动态代理

在运行时为指定的接口自动生成代理对象,并通过 invoke 方法增强了这些对象的功能

2. 两个核心组件

java.lang.reflect.Proxy

这个类提供了方法:newProxyInstance()用来创建一个代理对象

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

参数说明:
1. loader :用于加载代理类的类加载器;通常传入被代理类的类加载器
2. interfaces : 被代理类实现的一些接口,数组形式;
3. h : 实现了 InvocationHandler 接口的对象;

疑惑解释
【Q】为什么Loader是用于加载代理类的类加载器,但又传入的是被代理类的类加载器?
【A】由于动态代理类需要实现与被代理类相同的接口,那么就要代理类必须具有和被代理类完全相同的接口定义,要实现这一点需要代理类使用和被代理类相同的类加载器

java.lang.reflect.InvocationHandler接口(提供了invoke方法)

public interface InvocationHandler {/*** 当你使用代理对象调用方法的时候实际会调用到这个方法*/public Object invoke(Object proxy, Method method, Object[] args)throws Throwable;
}

参数说明:
1. proxy:动态生成的代理类
2. method : 与代理类对象调用的方法相对应
3. args : 当前 method 方法的参数

3. 使用示例(以简单的RPC 过程为例)

定义代理类ClientProxy

说明

  1. getProxy方法中,InvocationHandler h位置(第 3 个入口参数)处,要传实现了InvocationHandler接口的类。由于ClientProxy类实现了这个接口,所以这里在调用newProxyInstance直接传入了this
  2. invoke方法中,没有像别的动态代理示例那样,在其中调用method.invoke(target, args);。因为这里的场景是:在RPC过程中,客户端对输入参数的封装,然后发送给服务端,服务端来执行相应的方法(客户端只有接口定义,而没有接口实现,所以这里并不需要method.invoke(target, args))(但是在服务端中,会调用method.invoke(target, args)
package com.chanlee.crpc.v1.client;import com.chanlee.crpc.v1.common.RpcRequest;
import com.chanlee.crpc.v1.common.RpcResponse;
import lombok.AllArgsConstructor;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;import static com.chanlee.crpc.v1.client.IOClient.sendRequest;@AllArgsConstructor
public class ClientProxy implements InvocationHandler {/*** 服务端 IP*/private String host;/*** 服务端端口号*/private int port;public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//构建request请求RpcRequest request = RpcRequest.builder().interfaceName(method.getDeclaringClass().getName()).method(method.getName()).paramsTypes(method.getParameterTypes()).params(args).build();//发送请求并获取响应RpcResponse<Object> response = sendRequest(host, port, request);//返回结果数据return response.getData();}public <T> T getProxy(Class<T> tClass){Object o = Proxy.newProxyInstance(tClass.getClassLoader(),new Class[]{tClass},this);return (T)o;}
}

客户端代码(使用上面的代理类)

说明:

  1. 下方代码没有出现Proxy.newProxyInstance(...)是因为ClientProxy类中的getProxy(...)已经封装了该方法
  2. 当代理类调用对应的方法时,会被invoke()方法截取,然后执行invoke()方法中的代码逻辑
package com.chanlee.crpc.v1.client;import com.chanlee.crpc.v1.common.User;
import com.chanlee.crpc.v1.service.UserService;import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.Socket;/*** 客户端代码*/
public class Client{public static void main(String[] args){ClientProxy clientProxy = new ClientProxy("127.0.0.1", 8003);UserService proxy = clientProxy.getProxy(UserService.class);//调用方法 1User user = proxy.getUserById(1);System.out.println(user);//调用方法 2User codingBoy = User.builder().age(25).id(32).realName("coding boy").build();Integer i = proxy.insertUser(codingBoy);System.out.println(i);}}
http://www.xdnf.cn/news/374.html

相关文章:

  • 突破速率瓶颈:毫米波技术如何推动 5G 网络迈向极限?
  • Flink介绍——实时计算核心论文之Kafka论文总结
  • 用 R 语言打造交互式叙事地图:讲述黄河源区生态变化的故事
  • 毕业论文超清pdf带标签导出
  • CANFD技术在新能源汽车通信网络中的应用与可靠性分析
  • 【文件操作与IO】详细解析文件操作与IO (二)
  • PFC 是什么?
  • GN ninja 工程化构建例程
  • 定时器复习DSP【2025/4/18】
  • 项目之在线OJ
  • 工作督导 | 具有边缘型人格障碍倾向的高危来访者,咨询师如何应对?
  • 2025年危化品安全员考试题库及答案
  • 物联网平台管理系统
  • Origin LabTalk
  • rLLM - 使LLM的强化学习民主化
  • 用于数学定理和逻辑推理的符号系统
  • 【TVM教程】microTVM TFLite 指南
  • 从零开始学Python游戏编程31-类3
  • AI 数字短视频系统AI数字人源码开发:开启短视频行业发展新维度​
  • AUTOSAR图解==>AUTOSAR_SWS_E2ETransformer
  • 图像分类标注小工具
  • ABAP OLE
  • 『前端样式分享』联系我们卡片式布局 自适应屏幕 hover动效 在wikijs中使用 (代码拿来即用)
  • 使用Gone MCP 组件编写MCP Server
  • 《系统分析师-第三阶段—总结(一)》
  • LUN Capacity(Blocks) 是什么意思
  • Java项目—— 拼图小游戏(进阶版)
  • 表征流体作用力的参数及其特性
  • 2025年MathorCup数学应用挑战赛【B题成品论文第二版】(免费分享)
  • Allure测试报告按测试终端和测试类型智能分类查看