为什么阿里巴巴禁止使用存储过程?-程序员宅基地

点击上方 "程序员小乐"关注, 星标或置顶一起成长

每天凌晨00点00分, 第一时间与你相约

每日英文

The most sad is, know you not, but you still reluctant to leave.

最让人难过的是,明知道你们没有以后,但你还是舍不得抽身离开。

每日掏心

生活中的许多苦难,这个世界不是所有的人都懂你。青春已然逝去,少年的时光也已经消失。

来自:杨洋的围脖啊 | 责编:乐乐

链接:segmentfault.com/a/1190000011138993

程序员小乐(ID:study_tech)第 856 次推文  图源:百度

往日回顾:程序员申请加班调休被HR拒绝:996是行规,不想加班就趁早走人!

     

   正文   

之所以有这个题目,我既不是故意吸引眼球,也不想在本文对存储过程进行教科书般论述。《阿里巴巴Java开发手册》是这样规定的:


再结合我最近项目中遇到的存储过程问题,所以今天我打算来聊一聊这个问题。

这事儿要从去年在武汉出差时一位同事的发问说起,问题是这样的:

我觉得存储过程挺好用的,你为什么不建议用呢

当时我好似胸有万言,但终究没用一个实在的例子回答同事,只是从结论上大侃一通,代码相对于SQL,复用、扩展、通用性都要更强。想必同事并不信服。

现在想来,我最近正碰到的问题,算是一个可以回答同事的例子吧。

最近项目中有个新需求,需要校验一个用户是否有Job,Certification,Disclosure这三个业务数据。

翻看了代码发现,系统的用户个人页面的C#代码调用了三个存储过程,去抓取用户的Job,Certification,Disclosure数据。

我的新需求,自然需要复用这三个存储过程,否则:

若每一处都写一次抓取数据的业务逻辑代码,若业务逻辑发生变化,难以追查和维护所有读取Job,Certification,Disclosure的SQL。

如果我在C#代码中调用这已有的三个存储过程,事情本该非常快就能结束。我也是这么做的。

但code reviewer认为,我的需求中,并不需要Job,Certification,Disclosure这三个业务对象的数据。我只是需要给定用户是否有Job,Certification,Disclosure而已。所以我应将是否有无Job,Certification,Disclosure的判断逻辑写在数据库,最终通过网络从数据库传到web服务器的仅是true或false,节省网络流量,这样最好不过了。

也对。除开网络性能,从接口设计的角度讲,接口的传入和返回值,都应是你本身需要的数据,不应带有大量不需要或者需要caller去预处理的数据。从接口语义表达就可知调用的目的,这样代码可读性也会有大大提高。

那就动手改。但没想到的是问题来了。

为了讲述问题,我简化代码,假设系统现有的存储过程如下:


CREATEPROCEDURE [dbo].[GetJobs]
(
@PersonId int,
@OrganizaitionId int
)
AS
BEGIN
SELECT JobId,JobName,JobType FROM Job WHERE PersonId = @PersonId AND OrganizaitionId = @OrganizaitionId
END

我在新的存储过程中调用它,我需要获得该person的jobs的数量,即GetJobs返回结果集的count。

为了实现这一目的,首先想到的是使用临时表,将返回结果集存入临时表,再对其进行count(*)的计数操作:

CREATEPROCEDURE [dbo].[MyProc]
(
@PersonId int,
@OrganizaitionId int,
)
AS
BEGIN
CREATETABLE#Temp(
PersonId int,
OrganizaitionId int
)

INSERTINTO#Temp EXEC dbo.GetJobs
@PersonId = @PersonId,
@ParentOrgId = @ParentOrgId

SELECTCOUNT(*) FROM#Temp
END

这种办法简单有效,但它存在严重的维护问题。未来如果被调用的存储过程的返回结果集字段有变动,那么MyProc中的临时表结构也需要随之变化。这是令人难以接受的。

那么将MyProc中的INSERT INTO换为SELECT INTO呢?很遗憾,答案是不行。SQL本身并不支持这种用法。

给现有存储过程GetJobs加output参数?本例中因为GetJobs已被其他多处代码或SQL scripts调用,所以对现有现有存储过程进行改动会有不小风险。

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

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.

最终我没能找到一种满意的办法,无奈之下我在新写的存储过程中将查询Jobs的语句写一了次。

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

天知道还有没有机会和那位前同事再讨论这一话题呢。

欢迎在留言区留下你的观点,一起讨论提高。如果今天的文章让你有新的启发,学习能力的提升上有新的认识,欢迎转发分享给更多人。

欢迎各位读者加入订阅号程序员小乐技术群,在后台回复“加群”或者“学习”即可。

猜你还想看

阿里、腾讯、百度、华为、京东最新面试题汇集

如果世界上只有一种数据结构,那么我选择 hash

SpringBoot + Redis + 注解 + 拦截器 实现接口幂等性校验

高并发之 API 接口,分布式,防刷限流,如何做?

关注订阅号「程序员小乐」,收看更多精彩内容

嘿,你在看吗

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/xiaoxiaole0313/article/details/105911533

智能推荐

module_platform_driver宏解析_module platform driver-程序员宅基地

文章浏览阅读792次。转自http://blog.csdn.net/richu123/article/details/51331081该函数实际是一个宏,它在include/Linux/platform_device.h中定义如下:宏定义中对应3个参数:1个是结构体,另外两个是函数,分别用于注册和注销结构体。[cpp] view plain copy _module platform driver

Java圣经-程序员宅基地

文章浏览阅读2.9k次。 Java开发的方法论是什么?一个站在开发软件最前沿的技术怎么和方法论联系在一起呢?这对读者来说是一个新鲜的概念,对使用Java作为开发语言的软件公司来说同样如此。因此,当2003年JCOE出现在中国软 _java圣经

FastDFS+Nginx双机热备环境搭建笔记_fastdfs 双中心-程序员宅基地

文章浏览阅读2k次。根据CSDN上面的多篇文章,结合自己的操作,搭建了一个双机环境,做好笔记,后期好查! 两台文件fastDFS文件服务器服务器A: 192.168.31.32服务器B: 192.168.31.33服务器A和B均部署 tracker 和 storage 服务环境说明1、fastDFS主服务器 主机名称:FastDFS01FastDFS+nginx-1.15.5+libevent..._fastdfs 双中心

PHP与RBAC设计思路讲解与源码-程序员宅基地

文章浏览阅读114次。在说权限管理前,应该先知道权限管理要有哪些功能:(1)、用户只能访问,指定的控制器,指定的方法(2)、用户可以存在于多个用户组里(3)、用户组可以选择,指定的控制器,指定的方法(4)、可以添加控制器和方法RBAC(Role-Based Access Control,基于角色的访问控制),就是用户通过角色与权限进行关联。简单地说,一个用户拥有若干角色,每一个角色拥有若..._php rbac原码

HDC,CDC,CPaintDC,CClientDC,CWindowDC区别_cpaintdc 与hdc-程序员宅基地

文章浏览阅读821次。CDC是Windows绘图设备的基类CClientDC:(1)(客户区设备上下文)用于客户区的输出,与特定窗口关联,可以让开发者访问目标窗口中客户区,其构造函数中包含了GetDC,析构函数中包含了ReleaseDCCClientDC dc(this);CPen *pOldpen=dc.SelectObject(&pen);dc.MoveTo(m_ptOrigin);dc.L_cpaintdc 与hdc

Mysql基本语句_mysql语句-程序员宅基地

文章浏览阅读4.3w次,点赞30次,收藏189次。一、SQL概述结构化查询语言(Structured Query Language)简称SQL,是一种特殊目的的编程语言,是一种数据库查询和程序设计语言,用于存取数据以及查询、更新和管理关系数据库系统,同时也是数据库脚本文件的扩展名。从上可以看出我们数据库相关工作职位大概两种:DBA和DBDDBA是数据库管理员database administratorDBD是数据库开发人员database..._mysql语句

随便推点

Spring 学习笔记《注解》Spring Boot + SpringMVC + JSP + Mybatis 完整Demo_springboot + mybatis_ springmvc +jsp 操作手册-程序员宅基地

文章浏览阅读361次。Spring 学习笔记Spring 学习笔记《准备工作》Spring 学习笔记《依赖注入》—— 配置 applicationContext.xmlSpring 学习笔记《依赖注入》—— 扫描配置文件注入 Spring 学习笔记《依赖注入》—— 自动注入Spring 学习笔记《依赖注入》—— Bean 的作用域 scopeSpring 学习笔记《依赖注入》—— 注解注解..._springboot + mybatis_ springmvc +jsp 操作手册

新增provider时报错 Manifest merger failed with multiple errors, see logs_> manifest merger failed with multiple errors, see-程序员宅基地

文章浏览阅读934次,点赞2次,收藏4次。在AndroidManifest.xml中新增provider时报了错误: Manifest merger failed with multiple errors, see logs新增provider的代码:<provider android:name="android.support.v4.content.FileProvider" ..._> manifest merger failed with multiple errors, see logs myfileprovider

【雷达成像基础】---------从成像雷达开始说起(转载请标注德雅村支书)_多普勒波束锐化 原理-程序员宅基地

文章浏览阅读5.9k次,点赞6次,收藏16次。近期本人对于雷达的成像算法进行了一定的研读,从最基础的出发抽象成几个简单的问题,将在后面几天一一列出,:1.成像雷达技术中应用最广泛的是?答:合成孔径雷达(Synthetic Aperture Radar,SAR),主要应用的平台是机载和星载平台;它的成像质量非常高且易于实现;它通过在距离向上(径向)使用宽带信号,实现距离维度的高分辨,在方位向上则利用平台的移动实现空间上的等效长阵列,通过..._多普勒波束锐化 原理

CMU15445 2021-程序员宅基地

文章浏览阅读7k次,点赞9次,收藏63次。lab地址讲义地址《数据库系统概念》中文版有许多删减和错误,英文版看起来又费劲。数据密集型应用设计Google 开源项目风格指南Effective系列(Effective c++,Effective STL,Effective modern c++)由于刚开始不知道咋注册Gradescope,就只通过了本地的测试用例,做完了3个实验后才开始提交Gradescope。Gradescope的注册邀请码为4PR8G5,学校填Carnegie Mellon University就可以了。_15445

退出Docker Swarm集群模式_the docker engine you're using is running in swarm-程序员宅基地

文章浏览阅读7.7k次。退出Docker Swarm集群模式_the docker engine you're using is running in swarm mode

程序员的那些事儿 -- 高级程序员买衣服-程序员宅基地

文章浏览阅读637次。A是一个高级程序员,收入各方面在程序员队伍里属于是偏上上的,为此,A的自我感觉相当良好。一. 过程今天天气不错,心情也挺好的,冬天就要来了,A打算去买件衣服。于是,A进了一个名牌店,服务员B没有搭理A,而是不屑地看了A一眼。A顿时感觉到被羞辱了,怒火中烧…。于是,A走出去,进了隔壁另一个卖衣服的店子。第二家店子服务员C的态度就好多了,几分钟后,A买了三件。而最开始,A本来打算只买一件的。...