网易首页 > 网易号 > 正文 申请入驻

做 SQL 性能优化真是让人干瞪眼

0
分享至

  很多大数据计算都是用 SQL 实现的,跑得慢时就要去优化 SQL,但常常碰到让人干瞪眼的情况。

  比如,存储过程中有三条大概形如这样的语句执行得很慢:

  
selecta,b,sum(x) fromT groupbya,b where…;selectc,d,max(y) fromT groupbyc,d where…;selecta,c,avg(y),min(z) fromT groupbya,c where…;

  这里的 T 是个有数亿行的巨大表,要分别按三种方式分组,分组的结果集都不大。

  分组运算要遍历数据表,这三句 SQL 就要把这个大表遍历三次,对数亿行数据遍历一次的时间就不短,何况三遍。

  这种分组运算中,相对于遍历硬盘的时间,CPU 计算时间几乎可以忽略。如果可以在一次遍历中把多种分组汇总都计算出来,虽然 CPU 计算量并没有变少,但能大幅减少硬盘读取数据量,就能成倍提速了。

  如果 SQL 支持类似这样的语法:

  
from T -- 数据来自 T 表selecta,b,sum(x) groupbya,b where… -- 遍历中的第一种分组selectc,d,max(y) groupbyc,d where… -- 遍历中的第二种分组selecta,c,avg(y),min(z) groupbya,c where…; -- 遍历中的第三种分组

  能一次返回多个结果集,那就可以大幅提高性能了。

  可惜, SQL 没有这种语法,写不出这样的语句,只能用个变通的办法,就是用 group a,b,c,d 的写法先算出更细致的分组结果集,但要先存成一个临时表,才能进一步用 SQL 计算出目标结果。SQL 大致如下:

  
createtableT_temp asselecta,b,c,d,sum(casewhen… thenx else0end) sumx,max(casewhen… theny elsenullend) maxy,sum(casewhen… theny else0end) sumy,count(casewhen… then1elsenullend) county,min(casewhen… thenz elsenullend) minzgroupbya,b,c,d;selecta,b,sum(sumx) fromT_temp groupbya,b where…;selectc,d,max(maxy) fromT_temp groupbyc,d where…;selecta,c,sum(sumy)/sum(county),min(minz) fromT_temp groupbya,c where…;

  这样只要遍历一次了,但要把不同的 WHERE 条件转到前面的 case when 里,代码复杂很多,也会加大计算量。而且,计算临时表时分组字段的个数变得很多,结果集就有可能很大,最后还对这个临时表做多次遍历,计算性能也快不了。大结果集分组计算还要硬盘缓存,本身性能也很差。

  还可以用存储过程的数据库游标把数据一条一条 fetch 出来计算,但这要全自己实现一遍 WHERE 和 GROUP 的动作了,写起来太繁琐不说,数据库游标遍历数据的性能只会更差!

  只能干瞪眼!

  TopN 运算同样会遇到这种无奈。举个例子,用 Oracle 的 SQL 写 top5 大致是这样的:

  
select* from(selectx fromT orderbyx desc) whererownum<=5

  表 T 有 10 亿条数据,从 SQL 语句来看,是将全部数据大排序后取出前 5 名,剩下的排序结果就没用了!大排序成本很高,数据量很大内存装不下,会出现多次硬盘数据倒换,计算性能会非常差!

  避免大排序并不难,在内存中保持一个 5 条记录的小集合,遍历数据时,将已经计算过的数据前 5 名保存在这个小集合中,取到的新数据如果比当前的第 5 名大,则插入进去并丢掉现在的第 5 名,如果比当前的第 5 名要小,则不做动作。这样做,只要对 10 亿条数据遍历一次即可,而且内存占用很小,运算性能会大幅提升。

  这种算法本质上是把 TopN 也看作与求和、计数一样的聚合运算了,只不过返回的是集合而不是单值。SQL 要是能写成这样:select top(x,5) from T 就能避免大排序了。

  然而非常遗憾,SQL 没有显示的集合数据类型,聚合函数只能返回单值,写不出这种语句!

  不过好在全集的 TopN 比较简单,虽然 SQL 写成那样,数据库却通常会在工程上做优化,采用上述方法而避免大排序。所以 Oracle 算那条 SQL 并不慢。

  但是,如果 TopN 的情况复杂了,用到子查询中或者和 JOIN 混到一起的时候,优化引擎通常就不管用了。比如要在分组后计算每组的 TopN,用 SQL 写出来都有点困难。Oracle 的 SQL 写出来是这样:

  
select* from(selecty,x,row_number() over(partitionbyy orderbyx desc) rn fromT)wherern<=5

  这时候,数据库的优化引擎就晕了,不会再采用上面说的把 TopN 理解成聚合运算的办法。只能去做排序了,结果运算速度陡降!

  假如 SQL 的分组 TopN 能这样写:

  
selecty,top(x,5) fromT groupbyy

  把 top 看成和 sum 一样的聚合函数,这不仅更易读,而且也很容易高速运算。

  可惜,不行。

  还是干瞪眼!

  关联计算也是很常见的情况。以订单和多个表关联后做过滤计算为例,SQL 大体是这个样子:

  
selecto.oid,o.orderdate,o.amountfromorders oleftjoincity ci ono.cityid = ci.cityidleftjoinshipper sh ono.shid=sh.shidleftjoinemployee e ono.eid=e.eidleftjoinsupplier su ono.suid=su.suidwhereci.state='New York'ande.title = 'manager'and...

  订单表有几千万数据,城市、运货商、雇员、供应商等表数据量都不大。过滤条件字段可能会来自于这些表,而且是前端传参数到后台的,会动态变化。

  SQL 一般采用 HASH JOIN 算法实现这些关联,要计算 HASH 值并做比较。每次只能解析一个 JOIN,有 N 个 JOIN 要执行 N 遍动作,每次关联后都需要保持中间结果供下一轮使用,计算过程复杂,数据也会被遍历多次,计算性能不好。

  通常,这些关联的代码表都很小,可以先读入内存。如果将订单表中的各个关联字段预先做序号化处理,比如将雇员编号字段值转换为对应雇员表记录的序号。那么计算时,就可以用雇员编号字段值(也就是雇员表序号),直接取内存中雇员表对应位置的记录,性能比 HASH JOIN 快很多,而且只需将订单表遍历一次即可,速度提升会非常明显!

  也就是能把 SQL 写成下面的样子:

  
selecto.oid,o.orderdate,o.amountfromorders oleftjoincity c ono.cid = c.# -- 订单表的城市编号通过序号 #关联城市表leftjoinshipper sh ono.shid=sh.# -- 订单表运货商号通过序号 #关联运货商表leftjoinemployee e ono.eid=e.# -- 订单表的雇员编号通过序号 #关联雇员表leftjoinsupplier su ono.suid=su.# -- 订单表供应商编号通过序号 #关联供应商表whereci.state='New York'ande.title = 'manager'and...

  可惜的是,SQL 使用了无序集合概念,即使这些编号已经序号化了,数据库也无法利用这个特点,不能在对应的关联表这些无序集合上使用序号快速定位的机制,只能使用索引查找,而且数据库并不知道编号被序号化了,仍然会去计算 HASH 值和比对,性能还是很差!

  有好办法也实施不了,只能再次干瞪眼!

  还有高并发帐户查询,这个运算倒是很简单:

  
selectid,amt,tdate,… fromTwhereid='10100'andtdate>= to_date('2021-01-10', 'yyyy-MM-dd')andtdate

  在 T 表的几亿条历史数据中,快速找到某个帐户的几条到几千条明细,SQL 写出来并不复杂,难点是大并发时响应速度要达到秒级甚至更快。为了提高查询响应速度,一般都会对 T 表的 id 字段建索引:

  
createindexindex_T_1 onT(id)

  在数据库中,用索引查找单个帐户的速度很快,但并发很多时就会明显变慢。原因还是上面提到的 SQL 无序理论基础,总数据量很大,无法全读入内存,而数据库不能保证同一帐户的数据在物理上是连续存放的。硬盘有最小读取单位,在读不连续数据时,会取出很多无关内容,查询就会变慢。高并发访问的每个查询都慢一点,总体性能就会很差了。在非常重视体验的当下,谁敢让用户等待十秒以上?!

  容易想到的办法是,把几亿数据预先按照帐户排序,保证同一帐户的数据连续存储,查询时从硬盘上读出的数据块几乎都是目标值,性能就会得到大幅提升。

  但是,采用 SQL 体系的关系数据库并没有这个意识,不会强制保证数据存储的物理次序!这个问题不是 SQL 语法造成的,但也和 SQL 的理论基础相关,在关系数据库中还是没法实现这些算法。

  那咋办?只能干瞪眼吗?

  不能再用 SQL 和关系数据库了,要使用别的计算引擎。

  开源的计算器 SPL 基于创新的理论基础,支持更多的数据类型和运算,能够描述上述场景中的新算法。用简单便捷的 SPL 写代码,在短时间内能大幅提高计算性能!

  上面这些问题用 SPL 写出来的代码样例如下:

  一次遍历计算多种分组

  
A B 1 A1=file("T.ctx").open().cursor(a,b,c,d,x,y,z) 2 cursor A1 =A2.select(…).groups(a,b;sum(x)) 3
//定义遍历中的第一种过滤、分组 4 cursor =A4.select(…).groups(c,d;max(y)) 5
//定义遍历中的第二种过滤、分组 6 cursor =A6.select(…).groupx(a,c;avg(y),min(z)) 7
//定义遍历中的第三种过滤、分组 8 … //定义结束,开始计算三种方式的过滤、分组

  用聚合的方式计算 Top5

  全集 Top5(多线程并行计算)

  
A 1 =file("T.ctx").open() 2 =A1.cursor@m(x).total(top(-5,x), top(5,x)) 3 // top(-5,x)计算出 x 最大的前 5 名,top(5,x) 是 x 最小的前 5 名。

  分组 Top5(多线程并行计算)

  
A 1 =file("T.ctx").open() 2 =A1.cursor@m(x,y).groups(y;top(-5,x), top(5,x))

  用序号做关联的 SPL 代码:

  系统初始化

  
A 2 >env(city,file("city.btx").import@b()),env(employee,file("employee.btx").import@b()),... 3 //系统初始化时,几个小表读入内存

  查询

  
A 1 =file("orders.ctx").open().cursor(cid,eid,…).switch(cid,city:#;eid,employee:#;…) 2 =A1.select(cid.state='New York' && eid.title=="manager"…) 3 //先序号关联,再引用关联表字段写过滤条件

  高并发帐户查询的 SPL 代码:

  数据预处理,有序存储

  
A B 1 =file("T-original.ctx").open().cursor(id,tdate,amt,…) 2 =A1.sortx(id) =file("T.ctx") 3 =B2.create@r(#id,tdate,amt,…).append@i(A2) 4 =B2.open().index(index_id;id)
5 //将原数据排序后,另存为新表,并为账号建立索引

  帐户查询

  
A B 1 =T.icursor(;id==10100 && tdate>=date("2021-01-10") && tdate 2 //查询代码非常简单

  除了这些简单例子,SPL 还能实现更多高性能算法,比如有序归并实现订单和明细之间的关联、预关联技术实现多维分析中的多层维表关联、位存储技术实现上千个标签统计、布尔集合技术实现多个枚举值过滤条件的查询提速、时序分组技术实现复杂的漏斗分析等等。

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
没油了!姚明老队友打了20年,赚2.8亿,场均1.2分真要退役了

没油了!姚明老队友打了20年,赚2.8亿,场均1.2分真要退役了

大西体育
2026-04-18 21:58:23
Claude Design连夜突袭,Figma市值瞬间蒸发!或抢走全球UI设计师饭碗

Claude Design连夜突袭,Figma市值瞬间蒸发!或抢走全球UI设计师饭碗

新智元
2026-04-18 12:55:26
睡一觉存款清零!多地紧急预警:凌晨0-4点,千万别这样放手机

睡一觉存款清零!多地紧急预警:凌晨0-4点,千万别这样放手机

记录生活日常阿蜴
2026-04-18 08:16:20
刘德华回应挚友潘宏彬去世,最后一面未见难掩悲伤,拒谈丧礼细节

刘德华回应挚友潘宏彬去世,最后一面未见难掩悲伤,拒谈丧礼细节

螃蟹吃瓜摊
2026-04-18 20:50:40
景甜为顶级富豪代Y生子!?

景甜为顶级富豪代Y生子!?

八卦疯叔
2026-04-18 09:48:56
一条烟、一瓶酒,2026年5月起,连茶叶盒都得过纪检这道关

一条烟、一瓶酒,2026年5月起,连茶叶盒都得过纪检这道关

周哥一影视
2026-04-17 19:29:59
林青霞亲口爆料:当年我全裸给他看,他却吓得扭头就跑!

林青霞亲口爆料:当年我全裸给他看,他却吓得扭头就跑!

达文西看世界
2026-03-04 15:07:30
男子爬上泰山“五岳独尊”石刻拍照,景区:将核查其身份进行处理

男子爬上泰山“五岳独尊”石刻拍照,景区:将核查其身份进行处理

扬子晚报
2026-04-17 12:09:40
伊朗高级官员:伊美有望在数日内达成一项初步协议

伊朗高级官员:伊美有望在数日内达成一项初步协议

每日经济新闻
2026-04-18 08:05:29
苏林刚夸完复兴号,后脚越南列车就着火:越南高铁到底差在哪?

苏林刚夸完复兴号,后脚越南列车就着火:越南高铁到底差在哪?

芳芳历史烩
2026-04-18 17:40:21
恩比德和娇妻近照,连续3年大伤,3亿合同打不了,超模妻子很性感

恩比德和娇妻近照,连续3年大伤,3亿合同打不了,超模妻子很性感

大西体育
2026-04-18 21:57:19
尴尬!湖北一美女称隔壁男子塞纸条索要微信,结果她直接挂网上了

尴尬!湖北一美女称隔壁男子塞纸条索要微信,结果她直接挂网上了

火山詩话
2026-04-18 14:42:35
意甲最新积分战报:那不勒斯爆冷,罗马1-1亚特兰大,帕尔马险胜

意甲最新积分战报:那不勒斯爆冷,罗马1-1亚特兰大,帕尔马险胜

足球狗说
2026-04-19 04:50:00
不会干,你就别干了!中国勒令马士基停运,巴拿马的反应很有意思

不会干,你就别干了!中国勒令马士基停运,巴拿马的反应很有意思

阿腩讲娱乐
2026-04-18 13:00:07
“穷人过生日的模板”,19岁男孩生日宴火了,家长费尽心思反被嘲

“穷人过生日的模板”,19岁男孩生日宴火了,家长费尽心思反被嘲

妍妍教育日记
2026-04-18 08:15:03
4月21日油价调整!不是近5毛,别被忽悠

4月21日油价调整!不是近5毛,别被忽悠

沙雕小琳琳
2026-04-18 20:41:21
37岁!杜兰特意外受伤!火箭即将大战湖人

37岁!杜兰特意外受伤!火箭即将大战湖人

篮球实战宝典
2026-04-18 22:11:20
惨遭清零!爱德华兹啊!四年3亿!这就没了?

惨遭清零!爱德华兹啊!四年3亿!这就没了?

篮球盛世
2026-04-18 12:50:12
狂砸 1.4 亿!克洛普给皇马下死命令,先签 3 大巨星,1 人必卖

狂砸 1.4 亿!克洛普给皇马下死命令,先签 3 大巨星,1 人必卖

澜归序
2026-04-18 05:59:49
CBA最新排名:广东离奇输球神助攻北京 辽宁不敌浙江跌出前8

CBA最新排名:广东离奇输球神助攻北京 辽宁不敌浙江跌出前8

狼叔评论
2026-04-18 22:00:14
2026-04-19 05:43:00
IT架构师联盟 incentive-icons
IT架构师联盟
IT架构实战分享
842文章数 7672关注度
往期回顾 全部

科技要闻

传Meta下月拟裁8000 大举清退人力为AI腾位

头条要闻

伊朗革命卫队向油轮开火 伊朗最高领袖发声

头条要闻

伊朗革命卫队向油轮开火 伊朗最高领袖发声

体育要闻

时隔25年重返英超!没有人再嘲笑他了

娱乐要闻

刘德华回应潘宏彬去世,拒谈丧礼细节

财经要闻

"影子万科"2.0:管理层如何吸血万物云?

汽车要闻

奇瑞威麟R08 PRO正式上市 售价14.48万元起

态度原创

手机
数码
本地
家居
公开课

手机要闻

荣耀600系列参数、外观全曝光

数码要闻

华为版的科技春晚来了!Pura 90/Pura X Max下周发:阵容豪华

本地新闻

12吨巧克力有难,全网化身超级侦探添乱

家居要闻

法式线条 时光静淌

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版