Home > Java > javaTutorial > Java handwriting an RPC framework

Java handwriting an RPC framework

Guanhui
Release: 2020-06-17 17:31:03
forward
3391 people have browsed it

Java handwriting an RPC framework

The RPC framework is called the remote calling framework. The core principle of its implementation is that the consumer uses a dynamic proxy to proxy an interface (Dynamic proxy based on JDK, of course if Using CGLib, you can directly use methods without interface classes). By adding network transmission programming, the transmission call interface method name and method parameters are obtained by the provider, and then through reflection, the method of the interface is executed, and then the reflection is executed. The results are sent back to the consumer through network programming.

Now let’s implement these concepts in turn. Here we do the simplest implementation. Network programming uses BIO. You can use Netty in Reactor mode to rewrite it in a way with better performance. The serialization and deserialization used in network transmission are also native to Java. Of course, such transmission bytes are relatively large and can be processed using Google's protoBuffer or kryo. This is just for convenience to explain the principle.

pom

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0"

         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

    <modelVersion>4.0.0</modelVersion>

 

    <groupId>com.guanjian</groupId>

    <artifactId>rpc-framework</artifactId>

    <version>1.0-SNAPSHOT</version>

     

    <build>

        <plugins>

            <plugin>

                <groupId>org.apache.maven.plugins</groupId>

                <artifactId>maven-compiler-plugin</artifactId>

                <version>3.7.0</version>

                <configuration>

                    <source>1.8</source>

                    <target>1.8</target>

                    <encoding>UTF-8</encoding>

                </configuration>

            </plugin>

        </plugins>

    </build>

</project>

Copy after login

First of all, of course, the interface and the interface method we want to call remotely.

1

2

public interface HelloService {

    String sayHello(String content);}

Copy after login

Interface implementation class

1

2

public class HelloServiceImpl implements HelloService {    public String sayHello(String content) {        return "hello," + content;    }

}

Copy after login

Dynamic proxy on the consumer side, If you write the provider and consumer in two projects, the provider side needs the above interface and implementation classes, while the consumer side only needs the above interface.

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

public class ConsumerProxy {

    /**

     * 消费者端的动态代理

     * @param interfaceClass 代理的接口类

     * @param host 远程主机IP

     * @param port 远程主机端口

     * @param <T>

     * @return

     */

    @SuppressWarnings("unchecked")

    public static <T> T consume(final Class<T> interfaceClass,final String host,final int port) {

        return (T) Proxy.newProxyInstance(interfaceClass.getClassLoader(),

                new Class[]{interfaceClass}, (proxy,method,args) -> {

                    //创建一个客户端套接字

                    Socket socket = new Socket(host, port);

                    try {

                        //创建一个对外传输的对象流,绑定套接字

                        ObjectOutputStream output = new ObjectOutputStream(socket.getOutputStream());

                        try {

                            //将动态代理的方法名写入对外传输的对象流中

                            output.writeUTF(method.getName());

                            //将动态代理的方法的参数写入对外传输的对象流中

                            output.writeObject(args);

                            //创建一个对内传输的对象流,绑定套接字

                            //这里是为了获取提供者端传回的结果

                            ObjectInputStream input = new ObjectInputStream(socket.getInputStream());

                            try {

                                //从对内传输的对象流中获取结果

                                Object result = input.readObject();

                                if (result instanceof Throwable) {

                                    throw (Throwable) result;

                                }

                                return result;

                            } finally {

                                input.close();

                            }

                        } finally {

                            output.close();

                        }

                    } finally {

                        socket.close();

                    }

                }

        );

    }

}

Copy after login

For information about JDK dynamic proxy, please refer to AOP principles and self-implementation. For BIO, please refer to the comparison between traditional IO and NIO.

Provider-side network transmission and remote calling services

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

public class ProviderReflect {

    private static final ExecutorService executorService = Executors.newCachedThreadPool();

 

    /**

     * RPC监听和远程方法调用

     * @param service RPC远程方法调用的接口实例

     * @param port 监听的端口

     * @throws Exception

     */

    public static void provider(final Object service,int port) throws Exception {

        //创建服务端的套接字,绑定端口port

        ServerSocket serverSocket = new ServerSocket(port);

        while (true) {

            //开始接收客户端的消息,并以此创建套接字

            final Socket socket = serverSocket.accept();

            //多线程执行,这里的问题是连接数过大,线程池的线程数会耗尽

            executorService.execute(() -> {

                try {

                    //创建呢一个对内传输的对象流,并绑定套接字

                    ObjectInputStream input = new ObjectInputStream(socket.getInputStream());

                    try {

                        try {

                            //从对象流中读取接口方法的方法名

                            String methodName = input.readUTF();

                            //从对象流中读取接口方法的所有参数

                            Object[] args = (Object[]) input.readObject();

                            Class[] argsTypes = new Class[args.length];

                            for (int i = 0;i < args.length;i++) {

                                argsTypes[i] = args[i].getClass();

 

                            }

                            //创建一个对外传输的对象流,并绑定套接字

                            //这里是为了将反射执行结果传递回消费者端

                            ObjectOutputStream output = new ObjectOutputStream(socket.getOutputStream());

                            try {

                                Class<?>[] interfaces = service.getClass().getInterfaces();

                                Method method = null;

                                for (int i = 0;i < interfaces.length;i++) {

                                    method = interfaces[i].getDeclaredMethod(methodName,argsTypes);

                                    if (method != null) {

                                        break;

                                    }

                                }

                                Object result = method.invoke(service, args);

                                //将反射执行结果写入对外传输的对象流中

                                output.writeObject(result);

                            } catch (Throwable t) {

                                output.writeObject(t);

                            } finally {

                                output.close();

                            }

                        } catch (Exception e) {

                            e.printStackTrace();

                        } finally {

                            input.close();

                        }

                    } finally {

                        socket.close();

                    }

                } catch (Exception e) {

                    e.printStackTrace();

                }

            });

        }

    }

}

Copy after login

Start the network listening and remote calling on the provider side

1

2

3

4

5

6

public class RPCProviderMain {

    public static void main(String[] args) throws Exception {

        HelloService service = new HelloServiceImpl();

        ProviderReflect.provider(service,8083);

    }

}

Copy after login

Start the dynamic proxy call of the consumer

1

2

3

4

5

6

7

8

9

10

public class RPCConsumerMain {

    public static void main(String[] args) throws InterruptedException {

        HelloService service = ConsumerProxy.consume(HelloService.class,"127.0.0.1",8083);

        for (int i = 0;i < 1000;i++) {

            String hello = service.sayHello("你好_" + i);

            System.out.println(hello);

            Thread.sleep(1000);

        }

    }

}

Copy after login

Running result

hello, hello _0
hello, hello_1
hello, hello_2
hello, hello_3
hello, hello_4
hello, hello_5

.....

If you want to extend it into a high-performance RPC framework with Netty ProtoBuffer, you can refer to the related writing methods of Netty integrating Protobuffer.

Recommended tutorial: "PHP"

The above is the detailed content of Java handwriting an RPC framework. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
source:oschina.net
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template