| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 大数据 -> 设计好接口的三十六个锦囊 -> 正文阅读 |
|
[大数据]设计好接口的三十六个锦囊 |
作为后端开发,不管是什么语言, 后端开发工程师,主要工作就是:如何把一个接口设计好。 今天就给大家介绍,设计好接口的 36 个锦囊。 1. 接口参数校验入参出参校验是每个程序员必备的基本素养。设计接口,必须先校验参数。 比如入参是否允许为空,入参长度是否符合预期长度。这个要养成习惯,日常开发中,很多低级 bug 都是不校验参数导致的。
出参也是,比如你定义的接口报文,参数是不为空的,但是你的接口返回参数没有做校验,因为程序某些原因,返回别人一个 2. 修改老接口时,注意接口的兼容性很多 bug 都是因为修改了对外老接口但是却不做兼容导致的。关键这个问题多数是比较严重的,可能直接导致系统发版失败。新手程序员很容易犯这个错误。 所以,如果你的需求是在原来接口上做修改,尤其这个接口是对外提供服务的话,一定要考虑接口兼容。 举个例子吧,比如 dubbo 接口,原本是只接收 A、B 参数,现在加了一个参数 C,就可以考虑这样处理:
3. 设计接口时,充分考虑接口的可扩展性要根据实际业务场景设计接口,充分考虑接口的可扩展性。 比如你接到一个需求:用户添加或者修改员工时,需要刷脸。那你是反手提供一个员工管理的提交刷脸信息接口呢?还是先思考:提交刷脸是不是通用流程呢?比如转账或者一键贴现需要接入刷脸的话,你是否需要重新实现一个接口呢?还是当前按业务类型划分模块,复用这个接口就好,保留接口的可扩展性。 如果按模块划分的话,未来如果其他场景比如一键贴现接入刷脸的话,不用再搞一套新的接口,只需要新增枚举,然后复用刷脸通过流程接口,实现一键贴现刷脸的差异化即可。 4. 接口考虑是否需要防重处理如果前端重复请求,你的逻辑如何处理?是不是考虑接口去重处理。 当然,如果是查询类的请求,其实不用防重。如果是更新修改类的话,尤其金融转账类的,就要过滤重复请求了。 简单点,你可以使用 Redis 防重复请求,同样的请求方,一定时间间隔内的相同请求,考虑是否过滤。当然,转账类接口,并发不高的话,推荐使用数据库防重表,以唯一流水号作为主键或者唯一索引。 5. 重点接口,考虑线程池隔离一些登录、转账交易、下单等重要接口,考虑线程池隔离。 如果你所有业务都共用一个线程池,有些业务出 bug 导致线程池阻塞打满的话,那就杯具了,所有业务都影响了。 因此进行线程池隔离,重要业务分配多一点的核心线程,就能更好保护重要业务。 6. 调用第三方接口要考虑异常和超时处理如果你调用第三方接口,或者分布式远程服务的话,需要考虑:
7. 接口实现考虑熔断和降级当前互联网系统一般都是分布式部署的。而分布式系统中经常会出现某个基础服务不可用,最终导致整个系统不可用的情况,这种现象被称为服务雪崩效应。 比如分布式调用链路
为了应对服务雪崩,常见的做法是熔断和降级。最简单是加开关控制,当下游系统出问题时,开关降级,不再调用下游系统。还可以选用开源组件 8. 日志打印好,接口的关键代码,要有日志保驾护航关键业务代码无论身处何地,都应该有足够的日志保驾护航。 比如:你实现转账业务,转个几百万,然后转失败了,接着客户投诉,然后你还没有打印到日志,想想那种水深火热的困境下,你却毫无办法。。。 那么,你的转账业务都需要哪些日志信息呢?至少,方法调用前,入参需要打印吧,接口调用后,需要捕获一下异常吧,同时打印异常相关日志,如下:
9. 接口的功能定义要具备单一性单一性是指接口做的事情比较单一、专一。比如一个登录接口,它做的事情就只是校验账户名密码,然后返回登录成功以及 其实这也是微服务一些思想,接口的功能单一、明确。比如订单服务、积分、商品信息相关的接口都是划分开的。将来拆分微服务的话,是不是就比较简便啦。 10. 接口有些场景,使用异步更合理举个简单的例子,比如实现一个用户注册的接口,用户注册成功时,发个邮件或者短信去通知用户。这个邮件或者发短信,就更适合异步处理。因为总不能一个通知类的失败,导致注册失败吧。 至于做异步的方式,简单的就是用线程池。还可以使用消息队列,就是用户注册成功后,生产者产生一个注册成功的消息,消费者拉到注册成功的消息,就发送通知。 不是所有的接口都适合设计为同步接口。比如你要做一个转账的功能,如果是单笔的转账,你是可以把接口设计同步。用户发起转账时,客户端再静静等待转账结果就好。如果是批量转账,一个批次一千笔,甚至一万笔的,你则可以把接口设计为异步。就是用户发起批量转账时,持久化成功就先返回受理成功。然后用户隔十分钟或者十五分钟再来查转账结果就好。又或者,批量转账成功后,再回调上游系统。 11. 优化接口耗时,远程串行考虑改并行调用假设我们设计一个 APP 首页的接口,它需要查用户信息、需要查 banner 信息、需要查弹窗信息等等。那是一个一个接口串行调,还是并行调用呢? 如果是串行一个一个查,比如查用户信息 200ms,查 banner 信息 100ms、查弹窗信息 50ms,那一共就耗时 12. 接口合并或者说考虑批量处理思想数据库操作或者是远程调用时,能批量操作就不要 for 循环调用。 一个简单例子,我们平时一个列表明细数据插入数据库时,不要在 for 循环一条一条插入,建议一个批次几百条,进行批量插入。同理远程调用也类似想法,比如你查询营销标签是否命中,可以一个标签一个标签去查,也可以批量标签去查,那批量进行,效率就更高嘛。
小伙伴们是否了解过 13. 接口实现过程中,恰当使用缓存哪些场景适合使用缓存?读多写少且数据时效要求越低的场景。 缓存用得好,可以承载更多的请求,提升查询效率,减少数据库的压力。
一般用 14. 接口考虑热点数据隔离性瞬时间的高并发,可能会打垮你的系统。可以做一些热点数据的隔离。比如业务隔离、系统隔离、用户隔离、数据隔离等。
15. 可变参数配置化,比如红包皮肤切换等假如产品经理提了个红包需求,圣诞节的时候,红包皮肤为圣诞节相关的,春节的时候,为春节红包皮肤等。 如果在代码写死控制,可有类似以下代码:
如果到了元宵节的时候,运营小姐姐突然又有想法,红包皮肤换成灯笼相关的,这时候,是不是要去修改代码了,重新发布了? 从一开始接口设计时,可以实现一张红包皮肤的配置表,将红包皮肤做成配置化。更换红包皮肤,只需修改一下表数据就好了。 当然,还有一些场景适合一些配置化的参数:一个分页多少数量控制、某个抢红包多久时间过期这些,都可以搞到参数配置化表里面。这也是扩展性思想的一种体现。 16. 接口考虑幂等性接口是需要考虑幂等性的,尤其抢红包、转账这些重要接口。最直观的业务场景,就是用户连着点击两次,你的接口有没有?hold 住。或者消息队列出现重复消费的情况,你的业务逻辑怎么控制? 回忆下,什么是幂等?
大家别搞混哈,防重和幂等设计其实是有区别的。防重主要为了避免产生重复数据,把重复请求拦截下来即可。而幂等设计除了拦截已经处理的请求,还要求每次相同的请求都返回一样的效果。不过呢,很多时候,它们的处理流程、方案是类似的。 接口幂等实现方案主要有 8 种:
17. 读写分离,优先考虑读从库,注意主从延迟问题我们的数据库都是集群部署的,有主库也有从库,当前一般都是读写分离的。 比如写入数据,肯定是写入主库,但是对于读取实时性要求不高的数据,则优先考虑读从库,因为可以分担主库的压力。 如果读取从库的话,需要考虑主从延迟的问题。 18. 接口注意返回的数据量,如果数据量大需要分页一个接口返回报文,不应该包含过多的数据量。 过多的数据量不仅处理复杂,并且数据量传输的压力也非常大。 如果数量实在是比较大,可以分页返回,如果是功能不相关的报文,那应该考虑接口拆分。 19. 好的接口实现,离不开 SQL 优化我们做后端的,写好一个接口,离不开 SQL 优化。 SQL 优化从以下这几个维度思考:
20. 代码锁的粒度控制好什么是加锁粒度呢?
我们写代码时,如果不涉及到共享资源,就没有必要锁住。这就好像你上卫生间,不用把整个家都锁住,锁住卫生间门就可以了。 比如,在业务代码中,有一个 ArrayList 因为涉及到多线程操作,所以需要加锁操作,假设刚好又有一段比较耗时的操作(代码中的 反例:
正例:
21. 接口状态和错误需要统一明确提供必要的接口调用状态信息。比如一个转账接口调用是成功、失败、处理中还是受理成功等,需要明确告诉客户端。如果接口失败,那么具体失败的原因是什么。这些必要的信息都必须要告诉给客户端,因此需要定义明确的错误码和对应的描述。同时,尽量对报错信息封装一下,不要把后端的异常信息完全抛出到客户端。 22. 接口要考虑异常处理实现一个好的接口,离不开优雅的异常处理。 对于异常处理,提十个小建议:
23. 优化程序逻辑优化程序逻辑这块还是挺重要的,也就是说,你实现的业务代码,如果是比较复杂的话,建议把注释写清楚。还有,代码逻辑尽量清晰,代码尽量高效。
反例:
正例:
当然,这只是一些很小的一个例子,还有很多类似的例子,需要大家开发过程中,多点思考的哈。 24. 接口实现过程中,注意大文件、大事务、大对象
25. 你的接口,需要考虑限流如果你的系统每秒扛住的请求是 1000,如果一秒钟来了十万请求呢?换个角度就是说,高并发的时候,流量洪峰来了,超过系统的承载能力,怎么办呢? 如果不采取措施,所有的请求打过来,系统 CPU、内存、Load 负载飚得很高,最后请求处理不过来,所有的请求无法正常响应。 针对这种场景,我们可以采用限流方案。就是为了保护系统,多余的请求,直接丢弃。 限流定义:
可以使用 Guava 的 26. 代码实现时,注意运行时异常(比如空指针、下标越界等)日常开发中,我们需要采取措施规避数组边界溢出、被零整除、空指针等运行时错误。类似代码比较常见:
应该采取措施,预防一下数组边界溢出。正例如下:
27. 保证接口安全性如果你的 API 接口是对外提供的,需要保证接口的安全性。保证接口的安全性有?token 机制和接口签名。 token 机制身份验证方案还比较简单的,就是:
接口签名的方式,就是把接口请求相关信息(请求报文,包括请求时间戳、版本号、appid 等),客户端私钥加签,然后服务端用公钥验签,验证通过才认为是合法的、没有被篡改过的请求。 除了加签验签和 token 机制,接口报文一般是要加密的。当然,用 https 协议是会对报文加密的。如果是我们服务层的话,如何加解密呢?
有时候,接口的安全性,还包括手机号、身份证等信息的脱敏。就是说,用户的隐私数据,不能随便暴露。 28. 分布式事务,如何保证
分布式事务的几种解决方案:
29. 事务失效的一些经典场景我们的接口开发过程中,经常需要使用到事务。所以需要避开事务失效的一些经典场景。
30. 掌握常用的设计模式把代码写好,还是需要熟练常用的设计模式,比如策略模式、工厂模式、模板方法模式、观察者模式等等。 设计模式,是代码设计经验的总结。使用设计模式可以可重用代码、让代码更容易被他人理解、保证代码可靠性。 31. 写代码时,考虑线性安全问题在高并发情况下,
32. 接口定义清晰易懂,命名规范我们写代码,不仅仅是为了实现当前的功能,也要有利于后面的维护。 说到维护,代码不仅仅是写给自己看的,也是给别人看的。 所以接口定义要清晰易懂,命名规范。 33. 接口的版本控制接口要做好版本控制。就是说,请求基础报文,应该包含 比如客户端 APP 某个功能优化了,新老版本会共存,这时候我们的 34. 注意代码规范问题注意一些常见的代码坏味道:
35. 保证接口正确性,其实就是保证更少的 bug保证接口的正确性,换个角度讲,就是保证更少的bug,甚至是没有bug。所以接口开发完后,一般需要开发自测一下。 然后的话,接口的正确性还体现在,多线程并发的时候,保证数据的正确性,等等。比如做一笔转账交易,扣减余额的时候,可以通过 CAS 乐观锁的方式保证余额扣减正确。 如果你是实现秒杀接口,得防止超卖问题吧。可以使用 Redis 分布式锁防止超卖问题。 36. 学会沟通,跟前端沟通,跟产品沟通我把这一点放到最后,学会沟通是非常非常重要的。 比如你开发定义接口时,一定不能上来就自己埋头把接口定义完了,需要跟客户端先对齐接口。遇到一些难点时,跟技术 leader 对齐方案。实现需求的过程中,有什么问题,及时跟产品沟通。 总之就是,开发接口过程中,一定要沟通好~ (捡田螺的小男孩) |
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 | -2024/11/23 23:19:02- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |