【RPC项目】1.从不代理到动态代理

本文介绍了从不使用动态代理的简单RPC调用到引入动态代理以降低耦合性的过程。首先展示了基础的远程调用实现,然后通过动态代理改进,降低了Stub类的硬编码。未来优化方向包括解决相同方法名导致的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

公共类

在这里插入图片描述

先放一些公共类:

package common.ServiceImpl;

import common.Hello;
import common.HelloService;
import lombok.extern.slf4j.Slf4j;

@Slf4j
public class HelloServiceImpl implements HelloService {

  @Override
  public String hello(Hello hello) {
      log.info("HelloServiceImpl2收到: {}.", hello.getMessage());
      String result = "Hello description is " + hello.getDescription();
      log.info("HelloServiceImpl2返回: {}.", result);
      return result;
  }
}
package common;

public interface HelloService {
    String hello(Hello hello);
}
package common;

import lombok.*;

import java.io.Serializable;

/**
 * @Author:Summer
 * @Data:2022/2/17 21:33
 */
@AllArgsConstructor
@NoArgsConstructor
@Getter
@Setter
@Builder
@ToString
public class Hello implements Serializable {
    private String message;
    private String description;
}

不用动态代理 demo1

下面最简陋的远程调用,客户端进行方法调用,然后服务端接收后本地调用并将结果返回。

package demo1;

import common.Hello;
import common.HelloService;
import common.ServiceImpl.HelloServiceImpl;


import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

public class Server {

    public static boolean running = true;

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        ServerSocket serverSocket = new ServerSocket(9999);
        while (running){
            Socket s = serverSocket.accept();
            process(s);
            s.close();
        }
        serverSocket.close();
    }

    public static void process(Socket s) throws IOException, ClassNotFoundException {
        InputStream is = s.getInputStream();
        OutputStream os = s.getOutputStream();
        ObjectInputStream ois = new ObjectInputStream(is);
        DataOutputStream dos = new DataOutputStream(os);

        Hello hello = (Hello) ois.readObject();
        HelloService service = new HelloServiceImpl();
        dos.writeUTF(service.hello(hello));
        dos.flush();
    }
}
package demo1;

import common.Hello;

import java.io.*;
import java.net.Socket;

public class Client {
    public static void main(String[] args) throws IOException {
        Hello hello = new Hello("111", "233");
        String describ = new Stub().hello(hello);
        System.out.println(describ);
    }
}
package demo1;

import common.Hello;

import java.io.ByteArrayOutputStream;
import java.io.DataInputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.net.Socket;

/**
    @Author:Summer
    @Data:2022/2/17 21:53
 */
public class Stub {
    public String hello(Hello hello) throws IOException {
        Socket s = new Socket("127.0.0.1",9999);
        ByteArrayOutputStream bao = new ByteArrayOutputStream();

        ObjectOutputStream oos = new ObjectOutputStream(bao);
        oos.writeObject(hello);

        s.getOutputStream().write(bao.toByteArray());
        s.getOutputStream().flush();

        DataInputStream dis = new DataInputStream(s.getInputStream());
        String describ = dis.readUTF();
        System.out.print(describ);
        s.close();
        dis.close();
        return describ;
    }
}

可以看到很多地方都是写死的,并且对于客户端来说应该是调用 HelloServiceImpl 类中的 hello 方法才是真正的远程调用。

下一步优化目标:

  1. 使用动态代理实现 Stub 的远程调用,返回代理生成的 HelloServiceImpl 类
  2. 从写 InvocationHandler 方法,将现在 Stub 类中的 hello 方法重构到 invoke 方法中

加入动态代理的 demo1

package demo2;

import common.HelloService;
import common.ServiceImpl.HelloServiceImpl;

import java.io.*;
import java.lang.reflect.Method;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @Author:Summer
 * @Data:2022/2/17 22:41
 */
public class Server2 {
    public static boolean running = true;

    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(9999);
        while (running){
            Socket s = serverSocket.accept();
            process(s);
            s.close();
        }
        serverSocket.close();
    }

    public static void process(Socket s) throws Exception {
        InputStream is = s.getInputStream();
        OutputStream os = s.getOutputStream();
        ObjectInputStream ois = new ObjectInputStream(is);
        DataOutputStream dos = new DataOutputStream(os);

        String methodName = ois.readUTF();
        Class[] params = (Class[]) ois.readObject();
        Object[] args = (Object[]) ois.readObject();

        HelloService service = new HelloServiceImpl();
        Method m = HelloService.class.getMethod(methodName, params);
        String description = (String) m.invoke(service, args);

        dos.writeUTF(description);
        dos.flush();
        dos.close();
    }
}
package demo2;

import common.Hello;
import common.HelloService;

/**
 * @Author:Summer
 * @Data:2022/2/17 22:41
 */
public class Client2 {
    public static void main(String[] args) {
        HelloService service = Stub2.getStub();
        String description = service.hello(new Hello("111","233"));
        System.out.print(description);
    }
}
package demo2;

import common.HelloService;

import java.io.DataInputStream;
import java.io.ObjectOutputStream;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.net.Socket;

/**
 1. @Author:Summer
 2. @Data:2022/2/17 22:41
 */
public class Stub2 {
    public static HelloService getStub(){
        InvocationHandler h = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                Socket s = new Socket("127.0.0.1",9999);
                ObjectOutputStream oos = new ObjectOutputStream(s.getOutputStream());

                String methodName = method.getName();
                Class[] params = method.getParameterTypes();
                oos.writeUTF(methodName);
                oos.writeObject(params);
                oos.writeObject(args);
                oos.flush();

                DataInputStream dis = new DataInputStream(s.getInputStream());
                String description = dis.readUTF();

                s.close();
                oos.close();
                return description;
            }
        };
        Object o = Proxy.newProxyInstance(HelloService.class.getClassLoader(), new Class[]{HelloService.class}, h);
        return (HelloService) o;
    }
}

使用动态代理后明显比不使用耦合性更低,但是仍然有许多问题需要优化。
下一步优化目标:

1.将请求规范化(现在如果有两个类有同样的方法就会出现问题)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值