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

1. 什么是动态代理

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

2. 两个核心组件

java.lang.reflect.Proxy

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

java 复制代码
 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方法)

java 复制代码
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)
java 复制代码
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()方法中的代码逻辑
java 复制代码
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);

        //调用方法 1
        User user = proxy.getUserById(1);
        System.out.println(user);
        
        //调用方法 2
        User codingBoy = User.builder()
                .age(25)
                .id(32)
                .realName("coding boy")
                .build();
        Integer i = proxy.insertUser(codingBoy);
        System.out.println(i);
    }

}
相关推荐
付朝鲜8 分钟前
用自写的jQuery库+Ajax实现了省市联动
java·前端·javascript·ajax·jquery
小赵面校招11 分钟前
Spring Boot整合MyBatis全攻略:原理剖析与最佳实践
java·spring boot·mybatis
曼岛_14 分钟前
[Java实战]Spring Boot 3 整合 Ehcache 3(十九)
java·spring boot·spring
意倾城15 分钟前
Spring Boot 配置文件敏感信息加密:Jasypt 实战
java·spring boot·后端
曼岛_15 分钟前
[Java实战]Spring Boot 3 整合 Apache Shiro(二十一)
java·spring boot·apache
火皇40516 分钟前
Spring Boot 使用 OSHI 实现系统运行状态监控接口
java·spring boot·后端
带刺的坐椅18 分钟前
Java Solon v3.3.0 发布(国产优秀应用开发基座)
java·spring·solon
不再幻想,脚踏实地19 分钟前
Spring Boot配置文件
java·数据库·spring boot
pedestrian_h27 分钟前
Spring AI 开发本地deepseek对话快速上手笔记
java·spring boot·笔记·llm·ollama·deepseek
诚丞成27 分钟前
BFS算法篇——从晨曦到星辰,BFS算法在多源最短路径问题中的诗意航行(上)
java·算法·宽度优先