百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

手写一个rpc远程调用服务demo(rpc调用webservice)

off999 2025-03-20 17:38 120 浏览 0 评论

前言


  • 因为公司业务需求,使用了K8S + istio进行服务部署和治理,没有使用常规的springclould技术栈(包括注册中心nacos和openfeign远程服务调用)。
  • 所以就自己开发了一个基于AOP实现的rpc远程调用服务模块。其实现原理实现和feign类似,都是通过远程调用方法的代理对象发送HTTP请求并返回结果。
  • 废话不多说,下面直接上代码

代码

  • 下图是demo模块划分,common是公共模块,demo-order和demo-user是模拟两个服务调用。

  • 定义一个标识为远程调用类的注解 @RpcService ,有点类似于feign的@FeignClient注解。

/**
*
* @AUTHOR ZRH
* @DATE 2021/4/10
*/
@Component
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface RpcService {

/**
* 远程服务名称
*/
String service();

/**
* 端口
*/
String port();
}


  • 定义两个标识远程调用接口请求方式注解 @get和@post,相当于@PostMapping和@GetMapping。

/**
*
* @AUTHOR ZRH
* @DATE 2021/4/10
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Post {

/**
* 接口路由
*
* @
return
*/
String value();
}

/**
*
* @AUTHOR ZRH
* @DATE 2021/4/10
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Get {

/**
* 接口路由
*
* @
return
*/
String value();
}

  • 然后定义一个AOP切面处理类 AopRpcHandler。只要远程调用接口方法上有注解@Post或者@Get,就会对方法进行代理方式请求。因为这里不需要原本远程调用方法的执行结果,所以这里直接使用@Around环绕切面,并且不需要执行原方法,所以直接使用JoinPoint 做参数接口(ProceedingJoinPoint继承自JoinPoint,里面多了两个阻塞方法proceed,用于获取原代理方法的执行结果)。
  • 通过代理对象获取到原方法的参数值,参数名,接口路由地址,接口请求方式,远程服务和端口等等。使用okhttp工具类发送代理请求,然后返回响应结果。

/**
* @AUTHOR ZRH
* @DATE 2021/4/10
*/
@Slf4j
@Aspect
@Component
public class AopRpcHandler {

private final static String HTTP =
"http://";

@Around(value =
"@annotation(post)")
public String aopPost(JoinPoint joinPoint, Post post) {
String result = null;
String url = null;
try {
RpcService rpcService = (RpcService) joinPoint.getSignature().getDeclaringType().getAnnotation(RpcService.class);
url = HTTP + rpcService.service() +
":" + rpcService.port() + "/" + post.value();
Object[] args = joinPoint.getArgs();
result = OkHttpUtils.post(url, JSON.toJSONString(args[0]));
} catch (Throwable throwable) {
log.error(
"服务调用异常,url = [{}]", url);
}
return result;
}

@Around(value =
"@annotation(get)")
public String aopGet(JoinPoint joinPoint, Get get) {
String result = null;
String url = null;
try {
RpcService rpcService = (RpcService) joinPoint.getSignature().getDeclaringType().getAnnotation(RpcService.class);
url = HTTP + rpcService.service() +
":" + rpcService.port() + "/" + get.value();

MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Parameter[] parameters = signature.getMethod().getParameters();
if (parameters != null && parameters.length > 0) {
Object[] args = joinPoint.getArgs();
int length = parameters.length;
url +=
"?";
for (int i = 0; i < length; i++) {
url += parameters[i] +
"=" + args[i];
if (i != length - 1) {
url +=
"&";
}
}
}
result = OkHttpUtils.get(url);
} catch (Throwable throwable) {
log.error(
"服务调用异常,url = [{}]", url);
}
return result;
}
}

  • 然后在demo-user服务中如果有远程调用场景,就创建一个远程调用类。使用注解@RpcService和@Post即可。方法中的返回值和返回类型可以自定义,比如一般项目中会有统一的响应结果。

/**
* @AUTHOR ZRH
* @DATE 2021/4/10 0010 1:06
*/
@RpcService(service =
"demo-order", port = "18002")
public class AopRpcDemo {

@Post(
"post")
public String post(String param) {
return "1";
}
}

  • 在demo-user服务中使用和正常调用接口一样。

/**
* @AUTHOR ZRH
* @DATE 2021/4/10 0010 0:42
*/
@RestController
public class DemoController {

@Autowired
private AopRpcDemo aopRpcDemo;

@PostMapping(
"post")
public String
post() {
String post = aopRpcDemo.post(
"zrh.post");
System.out.println(
"调用远程接口方法返回结= " + post);
return "ok";
}
}

  • 如果就这样把demo服务启动后,访问是访问不了的。因为在aop切面处理类中对http请求的URL没有通过域名而是通过服务名称拼接的。
  • 这里如果是基于注册中心和feign进行服务调用,那是没有问题,因为feign会通过服务名称到注册中心找到对应服务的地址进行请求远程接口。
  • 而这里因为没有使用注册中心,所以在window上需要增加hosts文件上的地址映射关系。在C:\Windows\System32\drivers\etc目录下的hosts文件增加。并在cmd控制台中使用ipconfig /flushdns刷新DNS内容。
  • @RpcService中的service写服务名而不写服务访问域名,是因为如果是多机集群部署,那么就可以使用服务名映射域名方式通过Nginx负载均衡进行转发请求。如果直接写服务访问域名就只能访问一个机子上的服务了。

  • 先看一下两个服务的配置文件和demo-order的接口



  • 服务启动后,访问http://localhost:18001/post,结果如下图:



  • 最后的结果和我们想要的结果一致。
  • 上面的demo是很简单的实现。如果读者想要在自己项目中使用此类技术栈,那需要考虑服务容错,服务发现,服务限流等等是否能兼容等。

最后

  • openfeign其实是可以独立和springboot进行使用的。先引入openfeign的maven包


org.springframework.cloud
spring-cloud-starter-openfeign
3.0.2

  • 然后在使用@FeignClient注解时,对url配置接口的访问地址,最后的执行结果和上述的结果是一样的。

/**
* @AUTHOR ZRH
* @DATE 2021/4/10 0010 1:15
*/
@FeignClient(name =
"demo-user", url = "demo-user:18001")
public interface UserFeign {

@PostMapping(
"hello")
String hello(@RequestBody String param);
}


————————————————

相关推荐

智能abc输入法电脑版(智能abc输入法免费下载)

要安装智能ABC输入法,首先需要在笔记本上打开浏览器,然后在搜索栏中输入“智能ABC输入法下载”,找到官方网站或者可信赖的第三方网站,点击下载并安装该输入法软件。安装完成后,在输入法设置中选择启用智能...

如何给电脑设置密码开机密码

1、点击左下角开始,选择控制面板!(有的可以右击我的电脑)2、然后在选择用户账户3、一般没有设置密码的需要设置administrative的管理员密码!也可以创建新的账户4、然后在选择创建密码5、然后...

系统盘制作u盘要多大(制作系统u盘要多少g)

u盘制作启动盘,8g空间足够了。随着WINDOWS系统的不断完善,操作系统本身文件也越来越大,因为操作系统集成了更多的设备驱动和补丁,但是就WINDOWS10系统来说,有8g的空间足够把U盘做成启动...

网吧电脑怎么关闭防火墙(网吧如何关掉防火墙)

1、首先,我们点击电脑桌面左下角的微软按钮,弹出的界面,我们找到windows系统,点击打开它,弹出的界面,我们点击控制面板;2、弹出的界面,我们点击WindowsDefender防火墙;3、之后我...

win7安装需要标准nvm(安装win7要求)

1、把操作系统的安装镜像用WINRAR软件全部解压。2、找一个U盘,不小于8GB,格式化为FAT32格式,把上一步解压的文件复制到U盘中。3、重启电脑,按F12,选择电脑当前从U盘启动,进可以进入安装...

win10不兼容32位软件(win10系统不兼容软件)

使用电脑管家更新下驱动就可以了。1、打开腾讯电脑管家,点击“工具箱”。2、在工具箱里找到“硬件检测”。3、在硬件检测里点击“驱动安装”。4、可以看到“安装状态”,如果是未安装可以直接点击安装。首先你...

win7的屏保设置在哪里(win7 如何设置屏保)

要设置屏保,按照以下步骤进行操作:1.点击桌面上空白处右键,选择“个性化”。2.在个性化窗口中,点击左侧菜单栏中的“屏幕保护程序”选项。3.在“屏幕保护程序”窗口中,可以选择系统提供的屏保样式。...

电脑重新分区后如何恢复数据

如果电脑从新分区后没有进行数据覆盖,那么数据恢复的可能性是存在的。但是需要注意的是,分区操作本身就会对硬盘进行一定程度的数据覆盖,因此恢复成功的可能性会受到影响。如果您需要恢复分区后的数据,可以尝试使...

手机连接电脑没反应只能充电

手机连接到电脑上,只显示充电状态,可能有以下原因。1.使用的USB线只是充电线。可以更换USB数据线。2.手机的USB连接,设置为仅充电了。设置为数据传输就行了。3.电脑中缺少手机使用的USB驱动。可...

百度手机卫士官网(百度手机卫士官网个人中心)

百度卫士  百度卫士是百度公司全新出品的简单可信赖的系统工具软件,集合了电脑加速、系统清理、木马查杀和软件管理功能,竭力为用户提供轻巧、快速、智能、纯净的产品体验。百度卫士郑重承诺:永久免费、不骚扰...

怎么自己安装win10系统(怎么自己安装win10系统)

 WindowS10安装系统教程方法很简单,建议使用U盘启动盘安装方法,找到一个8g的U盘,用老毛桃或大白菜启动程序制作成启动盘,将windowS10操作系统传入U盘内,插在电脑USB接...

win7升级win10需要产品密钥(windows7升级10要激活码吗)

1、通过win+R打开“运行”窗口,输入“cmd”打开命令提示窗口,输入slmgr/upk,卸载当初前操作系统密钥;  2、打开当前操作系统“系统盘”找内到“ESD”(升级助手缓存的windows...

华硕声卡驱动下载官网(华硕官方声卡驱动)

如果华硕笔记本声卡驱动消失了,可以尝试以下方法进行修复。首先,进入设备管理器,检查是否存在声卡设备,并进行更新驱动。如果没有找到声卡设备,可以在华硕官方网站上搜索相应型号的声卡驱动程序,并下载安装最新...

sd卡低级格式化(格式化 sd卡)

1.首先将装有内存卡的读卡器插到电脑主机USB接口中。2.接着点这里下载【内存卡低格工具llftool】工具。3.下载完成后,对其进行解压缩,然后双击运行其中的可执行程序。4.在打开的修复程序窗口中,...

怎么把一个u盘做成2个系统盘

1、U盘插入电脑,鼠标右键点击桌面的“此电脑”,选择管理。2、然后再选择“磁盘管理”。3、找到U盘的两个分区,选择一个不需要的分区,鼠标右键点击这个分区选择“删除卷”。4、删除完之后会出现一个未分配的...

取消回复欢迎 发表评论: