深圳市住房建设局网站,网站主机安全,手机优化游戏性能的软件,网络推广好做吗多少钱很多大数据计算都是用SQL实现的#xff0c;跑得慢时就要去优化SQL#xff0c;但常常碰到让人干瞪眼的情况。
比如#xff0c;存储过程中有三条大概形如这样的语句执行得很慢#xff1a; select a,b,sum(x) from T group by a,b where …; select c,d,max(y) from T grou…很多大数据计算都是用SQL实现的跑得慢时就要去优化SQL但常常碰到让人干瞪眼的情况。
比如存储过程中有三条大概形如这样的语句执行得很慢 select a,b,sum(x) from T group by a,b where …; select c,d,max(y) from T group by c,d where …; select a,c,avg(y),min(z) from T group by a,c where …;这里的T是个有数亿行的巨大表要分别按三种方式分组分组的结果集都不大。
分组运算要遍历数据表这三句SQL就要把这个大表遍历三次对数亿行数据遍历一次的时间就不短何况三遍。
这种分组运算中相对于遍历硬盘的时间CPU计算时间几乎可以忽略。如果可以在一次遍历中把多种分组汇总都计算出来虽然CPU计算量并没有变少但能大幅减少硬盘读取数据量就能成倍提速了。
如果SQL支持类似这样的语法
from T --数据来自T表 select a,b,sum(x) group by a,b where … --遍历中的第一种分组 select c,d,max(y) group by c,d where … --遍历中的第二种分组 select a,c,avg(y),min(z) group by a,c where …; --遍历中的第三种分组能一次返回多个结果集那就可以大幅提高性能了。
可惜 SQL没有这种语法写不出这样的语句只能用个变通的办法就是用group a,b,c,d的写法先算出更细致的分组结果集但要先存成一个临时表才能进一步用SQL计算出目标结果。SQL大致如下 create table T\_temp as select a,b,c,d, sum(case when … then x else 0 end) sumx, max(case when … then y else null end) maxy, sum(case when … then y else 0 end) sumy, count(case when … then 1 else null end) county, min(case when … then z else null end) minz group by a,b,c,d;select a,b,sum(sumx) from T\_temp group by a,b where …; select c,d,max(maxy) from T\_temp group by c,d where …; select a,c,sum(sumy)/sum(county),min(minz) from T\_temp group by a,c where …; 这样只要遍历一次了但要把不同的WHERE条件转到前面的case when里代码复杂很多也会加大计算量。而且计算临时表时分组字段的个数变得很多结果集就有可能很大最后还对这个临时表做多次遍历计算性能也快不了。大结果集分组计算还要硬盘缓存本身性能也很差。
还可以用存储过程的数据库游标把数据一条一条fetch出来计算但这要全自己实现一遍WHERE和GROUP的动作了写起来太繁琐不说数据库游标遍历数据的性能只会更差
只能干瞪眼
TopN运算同样会遇到这种无奈。举个例子用Oracle的SQL写top5大致是这样的 select \* from (select x from T order by x desc) where rownum5表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 (select y,x,row\_number() over (partition by y order by x desc) rn from T) where rn5这时候数据库的优化引擎就晕了不会再采用上面说的把TopN理解成聚合运算的办法。只能去做排序了结果运算速度陡降
假如SQL的分组TopN能这样写 select y,top(x,5) from T group by y把top看成和sum一样的聚合函数这不仅更易读而且也很容易高速运算。
可惜不行。
还是干瞪眼
关联计算也是很常见的情况。以订单和多个表关联后做过滤计算为例SQL大体是这个样子
select o.oid,o.orderdate,o.amount
from orders o left join city ci on o.cityid ci.cityid left join shipper sh on o.shidsh.shid left join employee e on o.eide.eid left join supplier su on o.suidsu.suid
where ci.stateNew York and e.titlemanager and ... 订单表有几千万数据城市、运货商、雇员、供应商等表数据量都不大。过滤条件字段可能会来自于这些表而且是前端传参数到后台的会动态变化。
SQL一般采用HASH JOIN算法实现这些关联要计算 HASH 值并做比较。每次只能解析一个JOIN有N个JOIN要执行N遍动作每次关联后都需要保持中间结果供下一轮使用计算过程复杂数据也会被遍历多次计算性能不好。
通常这些关联的代码表都很小可以先读入内存。如果将订单表中的各个关联字段预先做序号化处理比如将雇员编号字段值转换为对应雇员表记录的序号。那么计算时就可以用雇员编号字段值也就是雇员表序号直接取内存中雇员表对应位置的记录性能比HASH JOIN快很多而且只需将订单表遍历一次即可速度提升会非常明显
也就是能把SQL写成下面的样子
select o.oid,o.orderdate,o.amount
from orders o left join city c on o.cid c.# --订单表的城市编号通过序号#关联城市表 left join shipper sh on o.shidsh.# --订单表运货商号通过序号#关联运货商表 left join employee e on o.eide.# --订单表的雇员编号通过序号#关联雇员表 left join supplier su on o.suidsu.#--订单表供应商号通过序号#关联供应商表
where ci.stateNew York and e.titlemanager and ... 可惜的是SQL 使用了无序集合概念即使这些编号已经序号化了数据库也无法利用这个特点不能在对应的关联表这些无序集合上使用序号快速定位的机制只能使用索引查找而且数据库并不知道编号被序号化了仍然会去计算 HASH 值和比对性能还是很差
有好办法也实施不了只能再次干瞪眼
还有高并发帐户查询这个运算倒是很简单
select id,amt,tdate,… from T
where id10100 and tdate to\_date(2021-01-10,yyyy-MM-dd) and tdateto_date(2021-01-25,yyyy-mm-dd) and … p在T表的几亿条历史数据中快速找到某个帐户的几条到几千条明细SQL写出来并不复杂难点是大并发时响应速度要达到秒级甚至更快。为了提高查询响应速度一般都会对 T 表的 id 字段建索引
create index index_T_1 on T(id)在数据库中用索引查找单个帐户的速度很快但并发很多时就会明显变慢。原因还是上面提到的SQL无序理论基础总数据量很大无法全读入内存而数据库不能保证同一帐户的数据在物理上是连续存放的。硬盘有最小读取单位在读不连续数据时会取出很多无关内容查询就会变慢。高并发访问的每个查询都慢一点总体性能就会很差了。在非常重视体验的当下谁敢让用户等待十秒以上
容易想到的办法是把几亿数据预先按照帐户排序保证同一帐户的数据连续存储查询时从硬盘上读出的数据块几乎都是目标值性能就会得到大幅提升。
但是采用SQL体系的关系数据库并没有这个意识不会强制保证数据存储的物理次序这个问题不是SQL语法造成的但也和SQL的理论基础相关在关系数据库中还是没法实现这些算法。
那咋办只能干瞪眼吗
不能再用SQL和关系数据库了要使用别的计算引擎。
开源的集算器SPL基于创新的理论基础支持更多的数据类型和运算能够描述上述场景中的新算法。用简单便捷的SPL写代码在短时间内能大幅提高计算性能
上面这些问题用SPL写出来的代码样例如下
一次遍历计算多种分组
AB1file(“T.ctx”).open().cursor(a,b,c,d,x,y,z2cursor A1A2.select(…).groups(a,b;sum(x))3//定义遍历中的第一种过滤、分组4cursorA4.select(…).groups(c,d;max(y))5//定义遍历中的第二种过滤、分组6cursorA6.select(…).groupx(a,c;avg(y),min(z))7//定义遍历中的第三种过滤、分组8…//定义结束开始计算三种方式的过滤、分组
用聚合的方式计算Top5
全集Top5多线程并行计算
A1file(“T.ctx”).open()2A1.cursorm(x).total(top(-5,x),top(5,x))3//top(-5,x) 计算出 x 最大的前 5 名top(5,x) 是 x 最小的前 5 名。
分组Top5多线程并行计算
A1file(“T.ctx”).open()2A1.cursorm(x,y).groups(y;top(-5,x),top(5,x))
用序号做关联的SPL代码
系统初始化
A1env(city,file(“city.btx”).importb()),env(employee,file(“employee.btx”).importb()),…2//系统初始化时几个小表读入内存
查询
A1file(“orders.ctx”).open().cursor(cid,eid,…).switch(cid,city:#;eid,employee:#;…)2A1.select(cid.state“New York” eid.title“manager”…)3//先序号关联再引用关联表字段写过滤条件
高并发帐户查询的SPL代码
数据预处理有序存储
AB1file(“T-original.ctx”).open().cursor(id,tdate,amt,…)2A1.sortx(id)file(“T.ctx”)3B2.creater(#id,tdate,amt,…).appendi(A2)4B2.open().index(index_id;id)5//将原数据排序后另存为新表并为帐号建立索引
帐户查询
A1T.icursor(;id10100 tdatedate(“2021-01-10”) tdatedate(“2021-01-25”) …,index_id).fetch()2//查询代码非常简单
除了这些简单例子SPL还能实现更多高性能算法比如有序归并实现订单和明细之间的关联、预关联技术实现多维分析中的多层维表关联、位存储技术实现上千个标签统计、布尔集合技术实现多个枚举值过滤条件的查询提速、时序分组技术实现复杂的漏斗分析等等。
正在为SQL性能优化头疼的小伙伴们可以和我们一起探讨
http://www.raqsoft.com.cn/wx/Query-run-batch-ad.html
SPL资料
SPL官网SPL下载SPL源代码