13 Mar 2024 |
一、存在的问题
1.重载方法在当前的实现中还不支持,调用了会报错。
data:image/s3,"s3://crabby-images/f4095/f4095f88dfb1e1bd17bf6a0baedd1d7a88e0a181" alt="图片"
data:image/s3,"s3://crabby-images/2e673/2e673a4c3d1019ee1db64b4c9e6ff0f89209147a" alt="图片"
2.类型转换也还存在问题。
假设定义的接口如下,参数是float类型。
data:image/s3,"s3://crabby-images/1e104/1e104f18ea2d64b4bf00ed50398bd3a70666ba19" alt="图片"
data:image/s3,"s3://crabby-images/99374/993740be00658e28530b566ad0164754e55f8264" alt="图片"
在Provider端接受到的是一个Double类型,这是因为web应用接收的请求后处理的类型。
data:image/s3,"s3://crabby-images/03f72/03f725445ca9bb4a0d2d0b5d1fcd7fd3e09d37cd" alt="图片"
在反射调用的时候就会报错。
data:image/s3,"s3://crabby-images/ef694/ef6946223784f47adffe78e91ac59d8e7ade263e" alt="图片"
二、解决方法重载问题
在Provider端创建的时候使用完整的方法签名替换方法全限定名。
data:image/s3,"s3://crabby-images/c37cc/c37cc366c1bbc33ff89bdbacf6f2413879ac6f8b" alt="图片"
方法签名:方法名称+参数个数+参数类型
data:image/s3,"s3://crabby-images/06ce7/06ce76339d702dfecff8e6e1e0eb32f4dbd52711" alt="图片"
在Consumer端封装请求参数时,传入方法签名即可。
data:image/s3,"s3://crabby-images/7b295/7b295505426e3f7b6259fcc39ccc4d19b22445ee" alt="图片"
举个例子:
data:image/s3,"s3://crabby-images/0c474/0c474090e07076beb394bc7d9e0f9e1cb584840d" alt="图片"
三、解决参数类型转换
在Provider端进行反射之前,处理请求参数。
data:image/s3,"s3://crabby-images/aa054/aa0541a958181bb7b8987559af4a0c8947bf1cf9" alt="图片"
processArgs()方法负责处理请求每个请求参数,传入的参数和方法参数类型匹配处理。
data:image/s3,"s3://crabby-images/b7456/b745652fe780f091d7850144599f2b4e6a1d0fa1" alt="图片"
同理,在Consumer端需要对返回的结果类型进行参数处理。
data:image/s3,"s3://crabby-images/62a34/62a347d1c5845d809f39d80100f166cc9a62708a" alt="图片"
参数处理工具类是TypeUtils.cast():
1.兼容的父子类型不需要处理;
2.处理数组类型:是一个什么样的数组,对每个值进行处理;
3.处理Map:使用json序列化;
4.基本类型使用封装类型;
data:image/s3,"s3://crabby-images/399b5/399b525135326a65378048788f50b0edf104eabc" alt="图片"
四、各种类型的测试:
1.引用类型参数;
2.int类型参数;
3.重载方法;
4.无参方法;
5.int类型参数,String类型返回值;
6.本地方法;
7.int类型参数,int类型返回值;
8.无参方法,String类型返回值;
9.数组类型返回值;
10.数组参数,数组返回值
data:image/s3,"s3://crabby-images/fa095/fa09518ec7956564e21ffc64b34b39a313d9b097" alt="图片"
工程地址:https://github.com/midnight2104/midnight-rpc/tree/lesson3
11 Mar 2024 |
一、RPC的简化版原理如下图(核心是代理机制)。
data:image/s3,"s3://crabby-images/8ae6e/8ae6e889de985adf8f9a12c5a92a3031a032955d" alt="图片"
1.本地代理存根: Stub
2.本地序列化反序列化
3.网络通信
4.远程序列化反序列化
5.远程服务存根: Skeleton
6.调用实际业务服务
7.原路返回服务结果
8.返回给本地调用方
二、新建一个模块rpc-demo-consumer
data:image/s3,"s3://crabby-images/e8a7a/e8a7a799e41d0adb9449578d331568058cd74452" alt="图片"
主要的依赖是rpc-demo-api,即业务服务定义的api接口,rpc-core是核心依赖包。
data:image/s3,"s3://crabby-images/4881b/4881bd08e033503001b24c735e33c7f8cda0ba17" alt="图片"
三、启动方法
在启动方法中注入业务服务接口UserService和OrderService,使用的注解是@RpcConsumer,本文主要的功能就是如何实现这个注解。
在启动的时候会导入ConsumerConfig的配置类。
使用的ApplicationRunner来测试方法调用。
data:image/s3,"s3://crabby-images/608e6/608e6802cd807a01e018d0f67b7fd317c4a16d09" alt="图片"
四、ConsumerConfig用来启动消费者服务。
data:image/s3,"s3://crabby-images/4a849/4a8490e83f12e2bf65b5828801f7acb5c715a7ff" alt="图片"
五、ConsumerBootstrap实现了ApplicationContextAware接口,注入ApplicationContext,即Spring的应用上下文,获取bean。
data:image/s3,"s3://crabby-images/b5ed1/b5ed1f1ff9c9b3ce35e6da29f36c9acc45eb1f28" alt="图片"
在start()方法中,
1.获取所有的bean;
2.遍历每个bean,查找bean中是否有字段用了@RpcConsumer注解;
3.对被标记了的@RpcConsumer的字段进行遍历;
4.获取到该字段的类型,全限定名;
5.使用动态代理的方式创建消费者对象;
6.把代理对象赋值给空字段。
data:image/s3,"s3://crabby-images/0a913/0a9132ecd5d261114dc426e91699bea7ecaea3b7" alt="图片"
查找bean中的字段是否使用了@RpcConsumer注解,通过循环遍历的方式进行判断。不断向父类寻找,是因为bean本身可能被Spring进行了CGLIB增强,形成了一个代理类。
data:image/s3,"s3://crabby-images/30e89/30e89028b7c1cc9a0611dd095c482d6058fb0bf7" alt="图片"
使用Proxy创建代理对象。
data:image/s3,"s3://crabby-images/9c936/9c93656fe7a074807d504e0b5b9c08e72b092509" alt="图片"
六、代理对象RpcInvocationHandler中定义了远程调用方法的逻辑。
data:image/s3,"s3://crabby-images/4965d/4965d016eb49ab19dd8b5d7722555dc028b87126" alt="图片"
在invoke()方法中定义具体实现:1.判断是否本地方法,就不需要走远程调用;
2.封装请求参数;
3.发起远程调用;
4.处理调用结果。
data:image/s3,"s3://crabby-images/0369a/0369acfc358cc1980c111945ec528a1fbdbf0501" alt="图片"
本地方法判断:即Object类中方法不需要被代理,不需要发起远程调用请求。
data:image/s3,"s3://crabby-images/b19e5/b19e55baf9c5c1eb4b9bb12bed1cc667f6c9bc6b" alt="图片"
RpcRequest封装了远程调用消费者端请求的数据结构:接口全限定名称、方法名称、方法参数。
data:image/s3,"s3://crabby-images/a5db4/a5db47c6e00b78b23902a88592da825f3383472c" alt="图片"
通过post方法发起远程调用,使用的是OkHttpClient。
data:image/s3,"s3://crabby-images/7f396/7f396f39228bc96a695184d2f47653b8a14a4fd9" alt="图片"
RpcResponse定义了消费者端的响应体数据结构:状态、响应结果数据、异常对象。
data:image/s3,"s3://crabby-images/fe342/fe3421c0280e7cecd9a9ee8ec3ab51f0690250e4" alt="图片"
处理调用结果:如果成功,就对调用结果数据进行反序列化;如果失败就封装异常信息。
data:image/s3,"s3://crabby-images/ca369/ca369f200eb88cbb65473af90a0e1c4e0eba5855" alt="图片"
七、对各种情况的基本测试:
1.类对象参数的远程调用;2.本地方法的远程调用;3.基本参数类型的远程调用;4. String类型参数的远程调用;5. 另一个业务范围(订单服务)的远程调用;6. 嵌套注入的远程调用;7. 异常测试
data:image/s3,"s3://crabby-images/60538/60538bdb8ac01d24792c989ca2cc40a9337273b5" alt="图片"
还有哪些问题?
- 默认使用的Java的动态代理,还有Spring的切面代理,CGLI增强,ByteBuddy的字节码
- 当前还是localhost请求,后续应该是走网络请求。
- 基本类型参数并不完整,如果是Double类型参数可能会报错。
工程地址:https://github.com/midnight2104/midnight-rpc
10 Mar 2024 |
RPC的简化版原理如下图(核心是代理机制)。
data:image/s3,"s3://crabby-images/6a787/6a7871669e4240d1104fcbd738db15977d256f03" alt="图片"
1.本地代理存根: Stub
2.本地序列化反序列化
3.网络通信
4.远程序列化反序列化
5.远程服务存根: Skeleton
6.调用实际业务服务
7.原路返回服务结果
8.返回给本地调用方
注意处理异常。
RpcProvider的本地实现
1、工程结构
rpc-core是核心实现类
rpc-demo-api是定义实际业务服务的接口
rpc-demo-provider是业务接口实现类
data:image/s3,"s3://crabby-images/82812/828127729490fe34f76c67ed4119a22669088e5e" alt="图片"
2、RpcProvider在启动过程中把@RpcProvider标记的方法存到Map中去
在启动时加载配置
data:image/s3,"s3://crabby-images/afd75/afd753159523c15f9b49e32255cd74ac5660bbba" alt="图片"
在配置中创建启动类
data:image/s3,"s3://crabby-images/aad1f/aad1fe3349d26e36470c60df359fe254c6bc2d7d" alt="图片"
启动类在创建的过程,会将带有@RpcProvider注解的类存放到Map中。
@PostConstruct注解可以在bean的初始化后进行执行,刚好满足我们的常见(存储bean)
使用 ApplicationContextAware接口是为了获取Spring的应用上下文ApplicationContext,从里面获取bean。
data:image/s3,"s3://crabby-images/47c00/47c00d18b9c29b41885258006d64fb9ea76b70cc" alt="图片"
RpcProvider是自定义的一个注解,用来表示这个类是一个服务提供者。
data:image/s3,"s3://crabby-images/e1765/e176586ac9155d61e3fcfb7c4a594bf4e19e6e3f" alt="图片"
一个服务提供者,订单服务实现类。
data:image/s3,"s3://crabby-images/b71e6/b71e6e38e288cd61465533cdf76eeda54680052c" alt="图片"
3、在调用的时候通过方法名称找到应该调用的方法,通过反射完成调用。
发起一个http请求调用,获取订单信息。
data:image/s3,"s3://crabby-images/42740/427404d3beb8da6d786f553a930493ede2f89f6c" alt="图片"
RpcRequest封装请求参数,包括:接口名称、方法名、参数。
data:image/s3,"s3://crabby-images/b49e6/b49e62fa5d86a3a1097aa59553748117ee9fe784" alt="图片"
一个请求入口
data:image/s3,"s3://crabby-images/10ef4/10ef4bf93e3da474ee381151e1f0d05b608f96f2" alt="图片"
请求调用,根据服务全限定名(就是一个类的全名)去map存找对应的类,找到后,通过反射发起方法调用。
data:image/s3,"s3://crabby-images/2d2ea/2d2eaf8dec76c944cfa7e85ac71afa6ae7bfd1e0" alt="图片"
响应类RpcResponse统一封装结果,返回状态,响应结果数据。
data:image/s3,"s3://crabby-images/00aeb/00aebbab3928b0a13c65fc7dbdeb74d27fd76401" alt="图片"
实际的调用结果,就成功了。
data:image/s3,"s3://crabby-images/fc6c9/fc6c90f5f9a5cede6c94422a95fda492819792d0" alt="图片"
4、还有哪些问题?
如果一个接口有多个实现类怎么办?
错误消息怎么处理?
方法有多个重载方法怎么办?
工程地址:https://github.com/midnight2104/midnight-rpc