为什么阿里巴巴禁止使用存储过程?

沙海 2021年5月28日04:48:33Java评论96字数 2832阅读9分26秒阅读模式
摘要

速读摘要

速读摘要文章源自JAVA秀-https://www.javaxiu.com/26552.html

当时我好似胸有万言,但终究没用一个实在的例子回答同事,只是从结论上大侃一通,代码相对于SQL,复用、扩展、通用性都要更强。最近项目中有个新需求,需要校验一个用户是否有Job,Certification,Disclosure这三个业务数据。未来如果被调用的存储过程的返回结果集字段有变动,那么MyProc中的临时表结构也需要随之变化。但对于业务逻辑的通用方法,非常不推荐将其写在存储过程中,代码复用、扩展与客户端语言比,相差甚远。文章源自JAVA秀-https://www.javaxiu.com/26552.html

原文约 2965 | 图片 6 | 建议阅读 6 分钟 | 评价反馈文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?

点击关注 ? Java基基 文章源自JAVA秀-https://www.javaxiu.com/26552.html

收录于话题文章源自JAVA秀-https://www.javaxiu.com/26552.html

#Java基基文章源自JAVA秀-https://www.javaxiu.com/26552.html

110个文章源自JAVA秀-https://www.javaxiu.com/26552.html

点击上方“Java基基”,选择“设为星标”文章源自JAVA秀-https://www.javaxiu.com/26552.html

做积极的人,而不是积极废人!文章源自JAVA秀-https://www.javaxiu.com/26552.html

文章源自JAVA秀-https://www.javaxiu.com/26552.html

源码精品专栏文章源自JAVA秀-https://www.javaxiu.com/26552.html

 文章源自JAVA秀-https://www.javaxiu.com/26552.html

文章源自JAVA秀-https://www.javaxiu.com/26552.html

来源:sf.gg//a/1190000011138993文章源自JAVA秀-https://www.javaxiu.com/26552.html

大家好,我是基基!文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

之所以有这个题目,我既不是故意吸引眼球,也不想在本文对存储过程进行教科书般论述。最近项目中遇到的存储过程问题,让我想起了去年在武汉出差时一位同事的发问:文章源自JAVA秀-https://www.javaxiu.com/26552.html

我觉得存储过程挺好用的,为什么你不建议用?文章源自JAVA秀-https://www.javaxiu.com/26552.html

当时我好似胸有万言,但终究没用一个实在的例子回答同事,只是从结论上大侃一通,代码相对于SQL,复用、扩展、通用性都要更强。想必同事并不信服。文章源自JAVA秀-https://www.javaxiu.com/26552.html

现在想来,我最近正碰到的问题,算是一个可以回答同事的例子吧。文章源自JAVA秀-https://www.javaxiu.com/26552.html

最近项目中有个新需求,需要校验一个用户是否有Job,Certification,Disclosure这三个业务数据。文章源自JAVA秀-https://www.javaxiu.com/26552.html

翻看了代码发现,系统的用户个人页面的C#代码调用了三个存储过程,去抓取用户的Job,Certification,Disclosure数据。我的新需求,自然需要复用这三个存储过程,否则:文章源自JAVA秀-https://www.javaxiu.com/26552.html

若每一处都写一次抓取数据的业务逻辑代码,若业务逻辑发生变化,难以追查和维护所有读取Job,Certification,Disclosure的SQL。文章源自JAVA秀-https://www.javaxiu.com/26552.html

如果我在C#代码中调用这已有的三个存储过程,事情本该非常快就能结束。我也是这么做的。文章源自JAVA秀-https://www.javaxiu.com/26552.html

但code reviewer认为,我的需求中,并不需要Job,Certification,Disclosure这三个业务对象的数据。我只是需要给定用户是否有Job,Certification,Disclosure而已。所以我应将是否有无Job,Certification,Disclosure的判断逻辑写在数据库,最终通过网络从数据库传到web服务器的仅是true或false,节省网络流量,这样最好不过了。也对。除开网络性能,从接口设计的角度讲,接口的传入和返回值,都应是你本身需要的数据,不应带有大量不需要或者需要caller去预处理的数据。从接口语义表达就可知调用的目的,这样代码可读性也会有大大提高。文章源自JAVA秀-https://www.javaxiu.com/26552.html

那就动手改。但没想到的是问题来了。文章源自JAVA秀-https://www.javaxiu.com/26552.html

为了讲述问题,我简化代码,假设系统现有的存储过程如下:文章源自JAVA秀-https://www.javaxiu.com/26552.html

CREATE PROCEDURE [dbo].[GetJobs](    @PersonId int,    @OrganizaitionId int )ASBEGIN  SELECT JobId,JobName,JobType FROM Job WHERE PersonId = @PersonId AND OrganizaitionId = @OrganizaitionIdEND

我在新的存储过程中调用它,我需要获得该person的jobs的数量,即GetJobs返回结果集的count。文章源自JAVA秀-https://www.javaxiu.com/26552.html

为了实现这一目的,首先想到的是使用临时表,将返回结果集存入临时表,再对其进行count(*)的计数操作:文章源自JAVA秀-https://www.javaxiu.com/26552.html

CREATE PROCEDURE [dbo].[MyProc](    @PersonId int,    @OrganizaitionId int, )ASBEGIN  CREATE TABLE #Temp(    PersonId int,    OrganizaitionId int  )  INSERT INTO #Temp EXEC dbo.GetJobs    @PersonId = @PersonId,    @ParentOrgId = @ParentOrgId  SELECT COUNT(*) FROM #TempEND

这种办法简单有效,但它存在严重的维护问题。未来如果被调用的存储过程的返回结果集字段有变动,那么MyProc中的临时表结构也需要随之变化。这是令人难以接受的。文章源自JAVA秀-https://www.javaxiu.com/26552.html

那么将MyProc中的INSERT INTO换为SELECT INTO呢?很遗憾,答案是不行。SQL本身并不支持这种用法。文章源自JAVA秀-https://www.javaxiu.com/26552.html

给现有存储过程GetJobsoutput参数?本例中因为GetJobs已被其他多处代码或SQL scripts调用,所以对现有现有存储过程进行改动会有不小风险。文章源自JAVA秀-https://www.javaxiu.com/26552.html

我搜遍网络,一位MS MVP的大神的文章几乎总结了所有存储过程之间传递数据的方法: How to Share Data between Stored Procedures。他在文章中也无可奈何地说道文章源自JAVA秀-https://www.javaxiu.com/26552.html

Keep in mind that compared to languages such as C# and Java, Transact-SQL is poorly equipped for code reuse, why solutions in T‑SQL to reuse code are clumsier.文章源自JAVA秀-https://www.javaxiu.com/26552.html

最终我没能找到一种满意的办法,无奈之下我在新写的存储过程中将查询Jobs的语句写一了次。文章源自JAVA秀-https://www.javaxiu.com/26552.html

存储过程在很多场景时有其优势,比如性能。但对于业务逻辑的通用方法,非常不推荐将其写在存储过程中,代码复用、扩展与客户端语言比,相差甚远。也许终究能实现,但代价与风险比客户端语言要高,得不偿失。文章源自JAVA秀-https://www.javaxiu.com/26552.html

天知道还有没有机会和那位前同事再讨论这一话题呢。文章源自JAVA秀-https://www.javaxiu.com/26552.html

文章源自JAVA秀-https://www.javaxiu.com/26552.html

欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

已在知识星球更新源码解析如下:文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

为什么阿里巴巴禁止使用存储过程?文章源自JAVA秀-https://www.javaxiu.com/26552.html

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 20 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。文章源自JAVA秀-https://www.javaxiu.com/26552.html

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。文章源自JAVA秀-https://www.javaxiu.com/26552.html

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。文章源自JAVA秀-https://www.javaxiu.com/26552.html

文章源自JAVA秀-https://www.javaxiu.com/26552.html

文章有帮助的话,在看,转发吧。谢谢支持哟 (*^__^*)
文章源自JAVA秀-https://www.javaxiu.com/26552.html

阅读原文文章源自JAVA秀-https://www.javaxiu.com/26552.html

继续阅读
速蛙云 - 极致体验,强烈推荐!!!购买套餐就免费送各大视频网站会员!快速稳定、独家福利社、流媒体稳定解锁!速度快,全球上网、视频、游戏加速、独立IP均支持!基础套餐性价比很高!这里不多说,我一直正在使用,推荐购买:https://www.javaxiu.com/59919.html
weinxin
资源分享QQ群
本站是JAVA秀团队的技术分享社区, 会经常分享资源和教程; 分享的时代, 请别再沉默!
沙海
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定