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

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

off999 2025-03-20 17:38 28 浏览 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);
}


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

相关推荐

Python自动化脚本应用与示例(python自动化脚本教程)

Python是编写自动化脚本的绝佳选择,因其语法简洁、库丰富且跨平台兼容性强。以下是Python自动化脚本的常见应用场景及示例,帮助你快速上手:一、常见自动化场景文件与目录操作O批量重命名文件...

如何使用Python实现一个APP(如何用python做一个程序)

要使用Python实现一个APP,你可以选择使用一些流行的移动应用开发框架,如Kivy、PyQt或Tkinter。这里以Kivy为例,它是一个跨平台的Python框架,可以用于创建漂亮的图形用户界面(...

免费定时运行Python程序并存储输出文档的服务推荐

免费定时运行Python程序并存储输出文档的服务推荐以下是几种可以免费定时运行Python程序并存储输出结果的云服务方案:1.PythonAnywhere特点:提供免费的Python托管环境支持定时...

【Python程序开发系列】如何让python脚本一直在后台保持运行

这是我的第385篇原创文章。一、引言让Python脚本在后台持续运行,有几种常见的方式,具体方式可以根据你的系统环境和需求选择。二、Linux或macOS系统2.1使用nohup命令no...

运行和执行Python程序(运行python的程序)

一、Python是一种解释型的脚本编程语言,这样的编程语言一般支持两种代码运行方式:交互式编程在命令行窗口中直接输入代码,按下回车键就可以运行代码,并立即看到输出结果;执行完一行代码,你还可以继续...

Python 初学者指南:计算程序的运行时长

在编写Python程序时,了解程序的运行时长是一项很有用的技能。这不仅能帮助你评估代码的效率,还能在优化程序性能时提供关键的数据支持。对于初学者来说,计算程序运行时长其实并不复杂,接下来就让我们看...

pyest+appium实现APP自动化测试,思路全总结在这里

每天进步一点点,关注我们哦,每天分享测试技术文章本文章出自【码同学软件测试】码同学公众号:自动化软件测试码同学抖音号:小码哥聊软件测试01appium环境搭建安装nodejshttp://nodej...

血脉觉醒后,编程小白我是如何通过Deepseek和Trae轻松开发软件的

以下就是作为一个编程小白的我,是如何一步步开发软件的保姆级教程,请点赞收藏:第一步:打开#deepseek#(首先关闭深度思考和联网搜索)输入或复制你要让它做一个什么样软件的要求和提示词(你可以先用...

我用Deepseek+Trae写的python小软件,小白也能轻松用上模型啦!

利用AI大模型deepseek,搭配TraeCN,用半个小时做了一个本地Ollama安装部署和一键卸载的小工具,哈哈哈!感觉还不错#deepseek#一直想做一个本地Ollama安装部署和一键卸载...

在安卓设备上运行Python的方法(安卓能运行python吗)

技术背景在安卓设备上运行Python可以为开发者提供更多的开发选择和灵活性,能够利用Python丰富的库和简洁的语法来开发各种应用,如游戏、脚本工具等。然而,由于安卓系统原生不支持Python,需要借...

零基础小白,DeepSeek全自动编程,超详细提示词,一键生成软件!

我前面发表了文章,详细说了编程零基础小白,如何利用DeepSeek进行编程的全过程,感兴趣的可以去看看:DeepSeek全自动编程很多人不会写提示词,不知道怎么开始对话。话不多说,请先看下图中的对话,...

小白用DeepSeek+Python编写软件(用python制作软件)

周末无事,用DeepSeek生成全部代码,写了一个mp3音乐播放器,几分钟搞定,DeepSeek确实太强大了。我的提示语是这么写的:“请用Python语言写一个音乐播放器,支持常见音乐格式,我是Pyt...

零基础使用DeepSeek开发Windows应用程序,超简单超实用!

你敢相信,我居然用DeepSeek开发了一个能用的Windows软件!整个过程就像和学霸同桌组队做作业,我负责提需求,DeepSeek负责写代码改bug,全程碰到任何问题直接丢给DeepSeek即可。...

第二篇:如何安装Python并运行你的第一个程序

欢迎回到我的Python入门教程系列!在上一篇中,我们讨论了为什么Python是一门值得学习的编程语言。今天,我们将迈出第一步:安装Python并运行你的第一个程序。无论你是Windows、macOS...

Python 运行,带你找入口,快速读懂程序

有C或Java编程开发经验的软件开发者,初次接触python程序,当你想快速读懂python项目工程时,是否觉得python程序有些太过随意,让你看有些无所适从,进而有些茫然。这是...

取消回复欢迎 发表评论: