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

都说Feign是RPC,没有侵入性,为什么我的代码越来越像 C++

ztj100 2024-12-29 07:22 46 浏览 0 评论

1. 概览

随着 Spring Cloud 的流行性,Feign 已经成为 RPC 的事实标准,由于其构建与 Http 协议之上,对请求和返回值缺少规范约束,在日常开发过程中经常由于设计不当对系统造成一定的侵入性。比如,很多公司基于 Web 经验对 Feign 返回体进行了约束,大致要求如下:

  1. 所有的请求统一返回统一的 FeignResult
  2. FeignResult 中的 code 代表处理状态,msg 代表异常信息,data 代表返回结果
  3. 所有请求统一返回 200,详细处理状态存储于 code

看规范定义,可以断定其出自于 Web 开发规范,但在使用过程中却为系统增加了太多的模板代码。

1.1. 背景

基于 Web 规范的 Feign 开发,让我们又回到了 C++ 时代,每次进行调用后,第一件事就是对 code 进行判断,示例如下:

public boolean login(Long userId){
     FeignResult<User> userFeignResult = this.userFeignClient.getByUserId(userId);
     if (userFeignResult.getCode() != SUCCESS){
         throw new BizException(userFeignResult.getMsg());
     }
     User user = userFeignResult.getData();
     return user.checkPassword(userId);
}

在拿到异常code后,往往读取 msg,然后抛出 自定义异常,从而中断处理流程。这些代码分布在系统的各处,枯燥无味还降低了代码的可读性。

对此,更加怀念 Dubbo 的做法,在 Client 和 Server 实现异常的穿透,最大限度的模拟 接口调用,让开发人员从重复代码中释放出来。

1.2. 目标

实现 Client 和 Server 间的异常穿透,使用 Java Exception 替代 Error Code 码,降低繁琐的模板代码。

  1. 区分 Web 和 Feign 请求,只对 Feign 请求进行处理
  2. 正常返回结果,不受任何影响
  3. 异常返回结果,直接抛出异常,在异常中保存详细的 code 和 msg 信息
  4. 可自定义异常,Client 调用失败时,抛出自定义异常

2. 快速入门

2.1. 准备环境

首先,引入 lego starter,在pom中增加依赖:

<groupId>com.geekhalo.lego</groupId>
<artifactId>lego-demo</artifactId>
<version>0.1.16-feign-SNAPSHOT</version>

FeignClientConfiuration 将自动完成核心 Bean 的注册,主要包括:

  1. RpcRequestInterceptor。为 Feign 调用添加标记头
  2. RpcHandlerExceptionResolver。对 Feign 调用进行异常处理
  3. RpcErrorDecoder。Feign 调用发生异常后,从 RpcErrorResult 恢复异常
  4. SimpleRpcExceptionResolver。将 RpcErrorResult 转化为 RpcException

2.2. 编写测试 Feign

首先,定义一个标准的 FeignApi ,具体如下:

public interface TestFeignApi { 
    @PostMapping("/test/postData/{key}")
    void postData(@PathVariable String key, @RequestBody List<Long> data);
    @PostMapping("/test/postDataForError/{key}")
    void postDataForError(@PathVariable String key, @RequestBody List<Long> data);
    @GetMapping("/test/getData/{key}")
    List<Long> getData(@PathVariable String key);
    @GetMapping("/test/getDataForError/{key}")
    List<Long> getDataForError(@PathVariable String key);
}

基于 TestFeignApi 实现 TestFeignClient,具体如下:

@FeignClient(name = "testFeignClient", url = "http://127.0.0.1:9090")
public interface TestFeignClient extends TestFeignApi{
}

最后,构建 TestFeignApi 实现 TestFeignService,具体如下:

@RestController
public class TestFeignService implements TestFeignApi{
    @Getter(AccessLevel.PRIVATE)
    public Map<String, List<Long>> cache = Maps.newHashMap();
    public List<Long> getByKey(String key){
        return this.cache.get(key);
    }
    @Override
    public void postData(String key, List<Long> data) {
        this.cache.put(key, data);
    }
    @Override
    public void postDataForError(String key, List<Long> data) {
        throw new TestPostException();
    }
    @Override
    public List<Long> getData(String key) {
        return this.cache.get(key);
    }
    @Override
    public List<Long> getDataForError(String key) {
        throw new TestGetException();
    }
}

2.3. 编写测试代码

编写单元测试如下:

@SpringBootTest(classes = DemoApplication.class, webEnvironment = DEFINED_PORT)
class TestFeignClientTest {
    @Autowired
    private TestFeignClient testFeignClient;
    @Autowired
    private TestFeignService testFeignService;
    private String key;
    private List<Long> data;
    @BeforeEach
    void setUp() {
        this.key = String.valueOf(RandomUtils.nextLong());
        this.data = Arrays.asList(RandomUtils.nextLong(), RandomUtils.nextLong(), RandomUtils.nextLong(), RandomUtils.nextLong());
    }
    @AfterEach
    void tearDown() {
    }
    @Test
    void postData(){
        this.testFeignClient.postData(key, data);
        Assertions.assertEquals(data, this.testFeignService.getData(key));
    }
    @Test
    void postDataForError(){
        Assertions.assertThrows(RpcException.class, ()->{
            this.testFeignClient.postDataForError(key, data);
        });
    }
    @Test
    void getData(){
        this.testFeignClient.postData(key, data);
        List<Long> data = this.testFeignService.getData(key);
        Assertions.assertEquals(data, this.data);
        List<Long> ds = this.testFeignClient.getData(key);
        Assertions.assertEquals(ds, this.data);
    }
    @Test
    void getDataForError(){
        this.testFeignClient.getData(key);
        Assertions.assertThrows(RpcException.class, ()->{
            this.testFeignClient.getDataForError(key);
        });
    }
}

执行单元测试,顺利通过,从测试结果中我们可得:

  1. 对于正常调用 postData 和 getData 方法,调用成功返回预期结果
  2. 对于异常调用 postDataForError 和 getDataForError 方法,直接抛出 RpcException

2.4. 定制异常

定制异常需要两个组件配合使用:

  1. CodeBasedException。在 Service 端使用,用于提供异常 code 和 msg 信息;
  2. RpcExceptionResolver。在 Client 端使用,基于 RpcErrorResult 将 code 恢复为指定异常;

首先,创建 CustomException,具体如下:

public class CustomException extends RuntimeException
    implements CodeBasedException {
    public static final int CODE = 550;
    @Override
    public int getErrorCode() {
        return CODE;
    }
    @Override
    public String getErrorMsg() {
        return "自定义异常";
    }
}

CutomException 实现CodeBasedException 接口,并对 getErrorCode 和 getErrorMsg 两个方法进行重写。

然后,增加 CustomExceptionResolver,具体如下:

@Component
public class CustomExceptionResolver implements RpcExceptionResolver {
    @Override
    public Exception resolve(String methodKey, int status, String remoteAppName, RpcErrorResult errorResult) {
        if (errorResult.getCode() == CustomException.CODE){
            throw new CustomException();
        }
        return null;
    }
}

CustomExceptionResolver 实现 RpcExceptionResolver 接口,对 CustomException.CODE 进行特殊处理,直接返回 CustomException。

最后,编写方法抛出 CustomException,具体如下:

@Override
public void customException() {
    throw new CustomException();
}

最后,编写并运行单元测试,具体如下:

@Test
void customException(){
    Assertions.assertThrows(CustomException.class, ()->{
        this.testFeignClient.customException();
    });
}

可见,客户端在调用时指教抛出 CustomException,而非 RpcException。

3. 设计&扩展

image

为了对异常的管理,我们对 Feign 和 Spring MVC 的组件进行定制,包括:

  1. RpcRequestInterceptor 实现 RequestInterceptor 接口。拦截 Feign 调用,在请求 Header 中添加 Feign 标签,用以标记该请求来自 Feign 调用
  2. RpcHandlerExceptionResolver 实现 HandlerExceptionResolver 接口。对 Spring MVC 出现的异常进行拦截,将异常信息转换为 RpcErrorResult 进行返回
  3. RpcErrorDecoder 实现 ErrorDecoder 接口。当请求返回码非 200 时进行调用,将 RpcErrorResult 转换为 RpcException 直接抛出

整个处理流程如下:

  1. 客户端调用 FeignClient 向 Server 发出请求,RpcRequestInterceptor 在请求头上添加标记 FeignRpc= YES
  2. 请求被 Spring MVC 的前置分发器 DispatcherServlet 处理
  3. DispatcherServlet 基于 HttpMessageConverter 将请求转换为方法参数,并调用业务方法;
  4. 如果业务方法调用成功
    1. HttpMessageConverter 将结果转化为 Json 并返回给客户端;
    2. FeignClient 的 Decoder 将 Json 转化为最终结果返回给调用方;
    3. 调用方成功拿到正常返回值
  5. 如果业务方法调用失败,抛出异常
    1. 异常被 RpcHandlerExceptionResolver 拦截
    2. RpcHandlerExceptionResolver 将 Exception 转化为 RpcErrorResult 并返回给客户端
    3. 异常返回码被 FeignClient 的 RpcErrorDecoder 拦截
    4. RpcErrorDecoder 读取 RpcErrorResult,并将其封装为 RpcException 直接抛出
    5. 调用方捕获异常进行处理

4. 项目信息

项目仓库地址:https://gitee.com/litao851025/lego

项目文档地址:https://gitee.com/litao851025/lego/wikis/support/feign

相关推荐

人生苦短,我要在VSCode里面用Python

轻沉发自浅度寺量子位出品|公众号QbitAI在程序员圈子里,VisualStudioCode(以下简称VSCode)可以说是目前最火的代码编辑器之一了。它是微软出品的一款可扩展的轻量...

亲测可用:Pycharm2019.3专业版永久激活教程

概述随着2020年的到来,又有一批Pycharm的激活码到期了,各位同仁估计也是在到处搜索激活方案,在这里,笔者为大家收录了一个永久激活的方案,亲测可用,欢迎下载尝试:免责声明本项目只做个人学习研究之...

Python新手入门很简单(python教程入门)

我之前学习python走过很多的歧途,自学永远都是瞎猫碰死耗子一样,毫无头绪。后来心里一直都有一个做头条知识分享的梦,希望自己能够帮助曾经类似自己的人,于是我来了,每天更新5篇Python文章,喜欢的...

Pycharm的设置和基本使用(pycharm运行设置)

这篇文章,主要是针对刚开始学习python语言,不怎么会使用pycharm的童鞋们;我来带领大家详细了解下pycharm页面及常用的一些功能,让大家能通过此篇文章能快速的开始编写python代码。一...

依旧是25年最拔尖的PyTorch实用教程!堪比付费级内容!

我真的想知道作者到底咋把PyTorch教程整得这么牛的啊?明明在内容上已经足以成为付费教材了,但作者偏要免费开源给大家学习!...

手把手教你 在Pytorch框架上部署和测试关键点人脸检测项目DBFace

这期教向大家介绍仅仅1.3M的轻量级高精度的关键点人脸检测模型DBFace,并手把手教你如何在自己的电脑端进行部署和测试运行,运行时bug解决。01.前言前段时间DBFace人脸检测库横空出世,...

进入Python的世界02外篇-Pycharm配置Pyqt6

为什么这样配置,要开发带UI的python也只能这样了,安装过程如下:一安装工具打开终端:pipinstallPyQt6PyQt6-tools二打开设置并汉化点击plugin,安装汉化插件,...

vs code如何配置使用Anaconda(vscode调用anaconda库)

上一篇文章中(Anaconda使用完全指南),我们能介绍了Anaconda的安装和使用,以及如何在pycharm中配置Anaconda。本篇,将继续介绍在vscode中配置conda...

pycharm中conda解释器无法配置(pycharm配置anaconda解释器)

之前用的好好的pycharm正常配置解释器突然不能用了?可以显示有这个环境然后确认后可以conda正在配置解释器,但是进度条结束后还是不成功!!试过了pycharm重启,pycharm重装,anaco...

Volta:跨平台开发者的福音,统一前端js工具链从未如此简单!

我们都知道现在已经进入了Rust时代,不仅很多终端常用的工具都被rust重写了,而且现在很多前端工具也开始被Rust接手了,这不,现在就出现了一款JS工具管理工具,有了它,你可以管理多版本的js工具,...

开发者的福音,ElectronEgg: 新一代桌面应用开发框架

今天给大家介绍一个开源项目electron-egg。如果你是一个JS的前端开发人员,以前面对这项任务桌面应用开发在时,可能会感到无从下手,甚至觉得这是一项困难的挑战。ElectronEgg的出现,它能...

超强经得起考验的低代码开发平台Frappe

#挑战30天在头条写日记#开始进行管理软件的开发来讲,如果从头做起不是不可以,但选择一款免费的且经得起时间考验的低代码开发平台是非常有必要的,将大幅提升代码的质量、加快开发的效率、以及提高程序的扩展性...

一文带你搞懂Vue3 底层源码(vue3核心源码解析)

作者:妹红大大转发链接:https://mp.weixin.qq.com/s/D_PRIMAD6i225Pn-a_lzPA前言vue3出来有一段时间了。今天正式开始记录一下梗vue3.0.0-be...

Windows 11 + WSL2 打造轻量级 Linux 本地开发环境实战教程

一、前言...

基于小程序 DSL(微信、支付宝)的,可扩展的多端研发框架

Mor(发音为/mr/,类似more),是饿了么开发的一款基于小程序DSL的,可扩展的多端研发框架,使用小程序原生DSL构建,使用者只需书写一套(微信或支付宝)小程序,就可以通过Mor...

取消回复欢迎 发表评论: