点击上方蓝色“方志朋”,选择“设为星标”

回复“666”获取独家整理的学习资料!

最近,我们的线上环境出现了一个问题,线上代码在执行过程中抛出了一个IllegalArgumentException,分析堆栈后,发现最根本的的异常是以下内容:

java.lang.IllegalArgumentException: No enum constant com.a.b.f.m.a.c.AType.P_M

大概就是以上的内容,看起来还是很简单的,提示的错误信息就是在AType这个枚举类中没有找到P_M这个枚举项。

于是经过排查,我们发现,在线上开始有这个异常之前,该应用依赖的一个下游系统有发布,而发布过程中是一个API包发生了变化,主要变化内容是在一个RPC接口的Response返回值类中的一个枚举参数AType中增加了P_M这个枚举项。

但是下游系统发布时,并未通知到我们负责的这个系统进行升级,所以就报错了。

我们来分析下为什么会发生这样的情况。

问题重现

首先,下游系统A提供了一个二方库的某一个接口的返回值中有一个参数类型是枚举类型。

一方库指的是本项目中的依赖

二方库指的是公司内部其他项目提供的依赖

三方库指的是其他组织、公司等来自第三方的依赖

public interface AFacadeService {public AResponse doSth(ARequest aRequest);}public Class AResponse{private Boolean success;private AType aType;}public enum AType{P_T,A_B}

然后B系统依赖了这个二方库,并且会通过RPC远程调用的方式调用AFacadeService的doSth方法。

public class BService {@AutowiredAFacadeService aFacadeService;public void doSth(){ARequest aRequest = new ARequest();AResponse aResponse = aFacadeService.doSth(aRequest);AType aType = aResponse.getAType();}}

这时候,如果A和B系统依赖的都是同一个二方库的话,两者使用到的枚举AType会是同一个类,里面的枚举项也都是一致的,这种情况不会有什么问题。

但是,如果有一天,这个二方库做了升级,在AType这个枚举类中增加了一个新的枚举项P_M,这时候只有系统A做了升级,但是系统B并没有做升级。

那么A系统依赖的的AType就是这样的:

public enum AType{P_T,A_B,P_M}

而B系统依赖的AType则是这样的:

public enum AType{P_T,A_B}

这种情况下,在B系统通过RPC调用A系统的时候,如果A系统返回的AResponse中的aType的类型为新增的P_M时候,B系统就会无法解析。一般在这种时候,RPC框架就会发生反序列化异常。导致程序被中断。

原理分析

这个问题的现象我们分析清楚了,那么再来看下原理是怎样的,为什么出现这样的异常呢。

其实这个原理也不难,这类RPC框架大多数会采用JSON的格式进行数据传输,也就是客户端会将返回值序列化成JSON字符串,而服务端会再将JSON字符串反序列化成一个Java对象。

而JSON在反序列化的过程中,对于一个枚举类型,会尝试调用对应的枚举类的valueOf方法来获取到对应的枚举。

而我们查看枚举类的valueOf方法的实现时,就可以发现,如果从枚举类中找不到对应的枚举项的时候,就会抛出IllegalArgumentException

public static <T extends Enum<T>> T valueOf(Class<T> enumType, String name) {T result = enumType.enumConstantDirectory().get(name);if (result != null)return result;if (name == null)throw new NullPointerException("Name is null");throw new IllegalArgumentException("No enum constant " + enumType.getCanonicalName() + "." + name);}

关于这个问题,其实在《阿里巴巴Java开发手册》中也有类似的约定:

这里面规定"对于二方库的参数可以使用枚举,但是返回值不允许使用枚举"。这背后的思考就是本文上面提到的内容。

扩展思考

为什么参数中可以有枚举?

不知道大家有没有想过这个问题,其实这个就和二方库的职责有点关系了。

一般情况下,A系统想要提供一个远程接口给别人调用的时候,就会定义一个二方库,告诉其调用方如何构造参数,调用哪个接口。

而这个二方库的调用方会根据其中定义的内容来进行调用。而参数的构造过程是由B系统完成的,如果B系统使用到的是一个旧的二方库,使用到的枚举自然是已有的一些,新增的就不会被用到,所以这样也不会出现问题。

比如前面的例子,B系统在调用A系统的时候,构造参数的时候使用到AType的时候就只有P_T和A_B两个选项,虽然A系统已经支持P_M了,但是B系统并没有使用到。

如果B系统想要使用P_M,那么就需要对该二方库进行升级。

但是,返回值就不一样了,返回值并不受客户端控制,服务端返回什么内容是根据他自己依赖的二方库决定的。

但是,其实相比较于手册中的规定,我更加倾向于,在RPC的接口中入参和出参都不要使用枚举。

一般,我们要使用枚举都是有几个考虑:

  • 1、枚举严格控制下游系统的传入内容,避免非法字符。

  • 2、方便下游系统知道都可以传哪些值,不容易出错。

不可否认,使用枚举确实有一些好处,但是我不建议使用主要有以下原因:

  • 1、如果二方库升级,并且删除了一个枚举中的部分枚举项,那么入参中使用枚举也会出现问题,调用方将无法识别该枚举项。

  • 2、有的时候,上下游系统有多个,如C系统通过B系统间接调用A系统,A系统的参数是由C系统传过来的,B系统只是做了一个参数的转换与组装。这种情况下,一旦A系统的二方库升级,那么B和C都要同时升级,任何一个不升级都将无法兼容。

我其实建议大家在接口中使用字符串代替枚举,相比较于枚举这种强类型,字符串算是一种弱类型。

如果使用字符串代替RPC接口中的枚举,那么就可以避免上面我们提到的两个问题,上游系统只需要传递字符串就行了,而具体的值的合法性,只需要在A系统内自己进行校验就可以了。

为了方便调用者使用,可以使用javadoc的@see注解表明这个字符串字段的取值从那个枚举中获取。

public Class AResponse{private Boolean success;/***  @see AType */private String aType;}

对于像阿里这种比较庞大的互联网公司,随便提供出去的一个接口,可能有上百个调用方,而接口升级也是常态,我们根本做不到每次二方库升级之后要求所有调用者跟着一起升级,这是完全不现实的,并且对于有些调用者来说,他用不到新特性,完全没必要做升级。

还有一种看起来比较特殊,但是实际上比较常见的情况,就是有的时候一个接口的声明在A包中,而一些枚举常量定义在B包中,比较常见的就是阿里的交易相关的信息,订单分很多层次,每次引入一个包的同时都需要引入几十个包。

对于调用者来说,我肯定是不希望我的系统引入太多的依赖的,一方面依赖多了会导致应用的编译过程很慢,并且很容易出现依赖冲突问题。

所以,在调用下游接口的时候,如果参数中字段的类型是枚举的话,那我没办法,必须得依赖他的二方库。但是如果不是枚举,只是一个字符串,那我就可以选择不依赖。

所以,我们在定义接口的时候,会尽量避免使用枚举这种强类型。规范中规定在返回值中不允许使用,而我自己要求更高,就是即使在接口的入参中我也很少使用。

最后,我只是不建议在对外提供的接口的出入参中使用枚举,并不是说彻底不要用枚举,我之前很多文章也提到过,枚举有很多好处,我在代码中也经常使用。所以,切不可因噎废食。

当然,文中的观点仅代表我个人,具体是是不是适用其他人,其他场景或者其他公司的实践,需要读者们自行分辨下,建议大家在使用的时候可以多思考一下。

热门内容:
  • 订单系统设计思路

  • 超美观的 Vue+Element 开源后台管理 UI

  • Mybatis 使用的 9 种设计模式,真是太有用了

  • 你的登录接口真的安全吗?

最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡

求你了,不要再在对外接口中使用枚举类型了!相关推荐

  1. 求封闭曲线面积vc代码_圆锥曲线综合5个类型,逐一突破

    (一)求圆锥曲线方程 求圆锥曲线方程分为五个类型,求解策略一般有以下几种: ①几何分析+方程思想: ②设而不求+韦达定理 ③定义+数形结合: ④参数法+方程思想 类型1--待定系数法 待定系数法本质就 ...

  2. c语言中error c2109 是什么错,求c语言改错 error C2109: 下标要求数组或指针类型?...

    求c语言改错 error C2109: 下标要求数组或指针类型? /*魔方阵*/ #include int main() { int i,j,a,n; int arr[100][100]; print ...

  3. 参数方程求二阶导时候不能直接把y,x分别对t求二阶导然后再相除作为d²y/dx²的原因

    首先看为什么参数方程求导可以用dy/dx=dy/dt/dx/dt计算,下面是证明 再看二阶导数d2y/dx2,他的推导过程类似上面的证明过程 下面是图片

  4. 手机OTG 我的世界_别再给手机外接OTG键鼠玩刺激战场了:其实还能这样操作

    刺激战场怎么用键鼠操作?刺激战场OTG键盘怎么设置?因为玩刺激战场这类的吃鸡FPS确实手搓不是特别爽,特别是压枪的时候感觉压不住,因此不少玩家会选择给手机外接键鼠来玩,可是其实使用这种操作方式缺陷: ...

  5. matlab析取范式求主析取范式用电脑,(p∧q)∨r 求其主析取范式 再用主析取范式求主合取范式...

    共回答了21个问题采纳率:90.5% 主合取范式:若干个极大项的合取. 主析取范式:若干个极小项的析取. 例, 求公式(p∧q)∨r的主析取范式及主合取范式. 主析取范式: (p∧q)∨r (p∧q∧ ...

  6. 别再给手机外接OTG键鼠玩刺激战场了:其实还能这样操作

    刺激战场怎么用键鼠操作?刺激战场OTG键盘怎么设置?因为玩刺激战场这类的吃鸡FPS确实手搓不是特别爽,特别是压枪的时候感觉压不住,因此不少玩家会选择给手机外接键鼠来玩,可是其实使用这种操作方式缺陷: ...

  7. 枚举算法 C语言 试题,求代码:如图大体思路是,用枚举法,一个个的试,碰到非的情况从...

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 //修正了一下 #include #include #include #include #include #include using namespace ...

  8. Redis 分布式锁使用不当,酿成一个重大事故,超卖了100瓶飞天茅台!!!

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! 基于Redis使用分布式锁在当今已经不是什么新鲜事了. 本 ...

  9. 小姐姐都能看懂的Happens-before规则,你还愣住了?

    前言 众所周知的并发编程三大特性:原子性.可见性.有序性:但是这些特性的起源你知道吗? 可见性:正是由于CPU存在缓存,导致了变量修改的不可见性: 原子性:线程切换是基于CPU指令而不是高级语言中的一 ...

最新文章

  1. Oracle 分析及动态采样
  2. mysql使用bka_MySQL Batched Key Access (BKA)原理和设置使用方法举例
  3. 全球计算机视觉顶会CVPR 2019论文出炉:腾讯优图25篇论文入选
  4. OpenStack(四)——Nova组件
  5. 第三方工具监控java进程_前9个免费的Java进程监视工具以及如何选择一种
  6. 深度学习pytorch--线性回归(二)
  7. 工作155:首页样式调整第二次
  8. RabbitMQ初识
  9. instanceof应用场景 instanceof一般用于对象类型强制转换
  10. sinx/x在0到+∞的积分
  11. usb 键盘码表_电脑键盘对应的二进制码表
  12. ong拼音汉字_拼音ong到底怎么读?
  13. 软件工程(系统流程图讲解)
  14. CentOS的 Oracle 11g R2安装
  15. Cascade R-CNN: Delving into High Quality Object Detection(个人学习笔记)
  16. 使用Termux在安卓手机上搭建本地Git服务器
  17. 控制台报错:Unknown database ‘xxxxx‘
  18. php基础教程推荐,php基础教程-绝对推荐
  19. python课程设计,学了python后你可以做的案例(词云,折线图,饼图…)
  20. 安装声卡驱动报错,代码:0xe0000246

热门文章

  1. python : unindent dose not math any outer indentation level
  2. 一起学WPF系列(2):第一个WPF应用程序
  3. Go-技篇第一 技巧杂烩
  4. IOS_多线程_ASI_AFN_UIWebView
  5. STM32F103 与 STM32F407引脚兼容问题
  6. 青少年编程竞赛交流群第048次活动录播
  7. Datawhale组队学习周报(第038周)
  8. 刻意练习:Python基础 -- Task09. else 与 with 语句
  9. 技术图文:如何利用C# 实现 Prim 最小生成树算法?
  10. 基于opencv的简单视频处理类示例