微服务 API 设计的实践与思考总结
被遗忘的注释,注释通常描述了API的功能以及参数说明,以及如何接入,甚至给出简单示例,过于详细的注释会带来一定的反作用,例如因为新需求带来了内部逻辑的调整,但是由于未及时对API的注释进行更新,会给新接入的调用方带来潜在的风险。所以不仅仅需要为API提供完整清晰的注释,当内部逻辑变更时,作为开发人员通常也需要评估API层面的变更,包括注释。
接口数量持续膨胀,有很多原因带来接口数量的膨胀,可能是接口升级,但是旧接口无法直接下线,所以会提供一个功能类似的新接口;可能是新接管一个服务由于对业务不了解,面对新需求直接开发新接口;可能是接口分类划分不合理,或者数据模型混乱导致API划分混乱,出现API功能重复,最后导致一个场景多个API接口都可以满足,这样很明显是应该避免的。解决这些问题都需要建立在对业务充分理解的基础上,下文的设计原则会针对这类问题给出解决方案。
缺乏有效测试,很多开发人员往往忽略对于接口的测试,无论是内部逻辑细节的单元测试,还是接口层面的测试,都是服务健壮性的一个有效保证,如果无法对接口进行有效测试,不仅是不负责任的提现,而且还会经常被线上bug困扰。
简单:在面向对象设计原则中,第一条是单一职责原则,同样适用于API设计,我们的主体对象就是业务模型,API就是封装内部逻辑后对外界开放的功能。保证API的简单和职责单一,能够避免解决上文中提到的接口数量膨胀问题。那如何才能实现API职责单一,需要我们在定义接口时能够准确识别出接口之间的关联性和边界,对于API如何划分可以通过以下角度:
按照业务主体划分,不一样的业务主体采用不一样的接口类
查询类和修改类的接口分离;通常来说我们对于数据的查询场景远大于修改的场景,而且查询有多种多样的业务场景,对于数据的修改请求通常来源于业务后台人员对数据进行修改,此时的业务逻辑也通常会更加特殊(例如有很多额外数据校验),所以建议修改类和查询类API尽量分离,甚至可以将业务配置后台类查询和普通业务查询分离以至于能够适应各自的业务变更。
专注:一个单一接口的场景是基于业务抽象后专注于某一个场景并且互相不重合的,这样才能保证接口的粒度足够小,尤其是对于基础类服务,接口粒度的划分能保证接口是纯粹的且互相独立的,这样才不至于在需求变化是涉及过多接口的变动(除非是对业务模型有较大的调整),另外要说明的是,内部逻辑的业务数据模型(POJO类)和API数据模型(DTO)有时候出现差异,否则可能需要消费者理解服务的业务模型才能正确的使用接口,这就要求在API设计中开发人员需要明确应该提供哪些数据模型给消费者,在此前提下更加有助于我们保证单一接口的专注。
注释应该包含哪些;接口的使用场景,参数的说明,在接口类说明中可以给出接口文档链接地址,方便调用方查看
参数的说明;包含参数代表的含义,参数的类型按照Javadoc link规范,参数是否为空,特殊值说明
过期说明;如果接口已经过期,需要给出过期说明,对于 Java 来时就是@Deprecated注解,并给出切换接口说明,如果条件允许可以推动调用方进行接口迁移,后续对旧接口下线
多参数结构与单一参数类结构 通常来说,如果一个接口的参数小于三个,那么建议使用多参数接口,这样做到直观简洁 如果一个接口的参数较多而且后续可能经常出现变动,为了便于扩展和兼容,会将参数封装到一个类结构中,记得同样对每个字段给出完整的注释说明。
类复用噩梦 在单一参数类结构下,我经常看到多个存在明显功能差异的接口频繁复用一个结构体,甚至接口参数和返回值都复用一个DTO,为了保证兼容,又不得不在同一个DTO内不断加字段,久而久之维护成本持续增高,这是一种不合理的类设计,如果遵守专注原则,这个问题很多时候可以避免。
接口逻辑或者参数变更时,需要对旧的接口保持兼容,这个是API变更时一定要遵守的原则之一,而且要通过接口测试来验证兼容性。
是否要新增接口,当面对一个新的需求时,为了避免对旧接口直接修改,有的开发人员会统一提供新的接口,如果并非逻辑上发生较大的变更,这样做会提高API的维护成本,后续如果不对API进行重构,新增加的维护成本将远大于最开始节省的开发成本,例如需要对某个参数增加有效校验,那么我们需要对两个接口的API实现都做修改,而且是重复性的代码,而且我们的影响范围已经成了两个接口,这样影响范围的扩大也带来了更多的潜在风险。当然在某些场景例如接口逻辑出现大的调整,API重构等情况下,更好的方法是提供新的接口,并推动服务消费者使用新的API,最后慢慢下线旧的API,这样才能遵循简单和专注的原则。
单元测试,完善的单元测试能保证代码的健壮性,提前在编码阶段发现并解决潜在的bug,单元测试是一个开发人员的必备能力。
接口和场景测试,接口测试包含内部逻辑验证,异常输入,并发等场景下对单一接口的验证,如果要对API进行完整的逻辑验证,需要开发人员构造完整的测试数据(通常包含scheme.sql和data.sql文件),尤其是对于基础服务,需要对某些复杂业务场景下联合多个接口完成某个场景的测试,并对中间的数据和输出进行Assert确认,这样也会代码一定的测试代码维护成本,需要开发人员进行利弊权衡。
逆锋起笔
是一个专注于程序员圈子的技术平台,你可以收获最新技术动态
、最新内测资格
、BAT等大厂大佬的经验
、增长自身
、学习资料
、职业路线
、赚钱思维
,微信搜索逆锋起笔
关注!