当前位置:首页 > 技术分析 > 正文内容

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

ruisui883个月前 (02-03)技术分析43

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

扫描二维码推送至手机访问。

版权声明:本文由ruisui88发布,如需转载请注明出处。

本文链接:http://www.ruisui88.com/post/1197.html

标签: feign 使用
分享给朋友:

“都说Feign是RPC,没有侵入性,为什么我的代码越来越像 C++” 的相关文章

HTML5+眼球追踪?黑科技颠覆传统手机体验

今天,iH5工具推出一个新的神秘功能——眼动追踪,可以通过摄像头捕捉观众眼球活动!为了给大家具体演示该功能的使用,我做了一个案例,供大家参考。实际效果如下:案例比较简单,就是通过眼动功能获取视觉焦点位置,剔除用户看中的牌。现在,舞台的属性中多了一个“启用眼动”的选项,另外,还多了一个“启用摄像头”的...

Gemini应用在Android上广泛推出2.0闪电模式切换器

#头条精品计划# 快速导读谷歌(搜索)应用的测试频道在安卓设备的双子应用中推出了2.0闪电实验功能,现已向稳定用户开放。双子应用通过谷歌应用运行,目前推出的15.50版本中,用户可通过模型选择器体验不同选项,包括1.5专业版、1.5闪电版和2.0闪电实验版。2.0闪电实验模型提供了更快的响应速度和优...

学前端,这30个CSS选择器,你必须熟记

你学会了基本的id,class类选择器和descendant后代选择器,然后就觉得完事了吗?如果这样,你就会错过许多灵活运用CSS的机会。虽然本文提到的许多选择器都属于CSS3,并且只能在现代的浏览器中使用,但学会这些是大有好处的。什么是CSS选择器呢?每一条css样式定义由两部分组成,形式如下:[...

数组、去重、排序、合并、过滤、删除

ES6数字去重 Array.from(new Set([1,2,3,3,4,4])) //[1,2,3,4] [...new Set([1,2,3,3,4,4])] //[1,2,3,4]2、ES6数字排序 [1,2,3,4].sort(); // [1, 2,3,4],默认是升序...

vue中router常见的三种传参方式

目录:我们在使用vue开发的过程中使用router跳转的时候肯定会遇到传参的情况;一般情况就三种传参是最常见的;那我们就来看看都有那几种传参方式吧!第一种:{ path: '/mall:id', name: 'Mall', component:...

VUE3+JAVA商城源码小程序APP商城

三勾小程序商城基于springboot+element-ui+uniapp打造的面向开发的小程序商城,方便二次开发或直接使用,可发布到多端,包括微信小程序、微信公众号、QQ小程序、支付宝小程序、字节跳动小程序、百度小程序、android端、ios端。软件架构后端: springboot2.3.12管...