幼儿饰品瑜伽美体用品微软
投稿投诉
微软创意
爱情通信
用品婚姻
爱好看病
美体软件
影音星座
瑜伽周边
星座办公
饰品塑形
搞笑减肥
幼儿两性
智家潮品

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

  很多大数据计算都是用SQL实现的,跑得慢时就要去优化SQL,但常常碰到让人干瞪眼的情况。
  比如,存储过程中有三条大概形如这样的语句执行得很慢:selecta,b,sum(x)fromTgroupbya,bwhere;selectc,d,max(y)fromTgroupbyc,dwhere;selecta,c,avg(y),min(z)fromTgroupbya,cwhere;
  这里的T是个有数亿行的巨大表,要分别按三种方式分组,分组的结果集都不大。
  分组运算要遍历数据表,这三句SQL就要把这个大表遍历三次,对数亿行数据遍历一次的时间就不短,何况三遍。
  这种分组运算中,相对于遍历硬盘的时间,CPU计算时间几乎可以忽略。如果可以在一次遍历中把多种分组汇总都计算出来,虽然CPU计算量并没有变少,但能大幅减少硬盘读取数据量,就能成倍提速了。
  如果SQL支持类似这样的语法:fromT数据来自T表selecta,b,sum(x)groupbya,bwhere遍历中的第一种分组selectc,d,max(y)groupbyc,dwhere遍历中的第二种分组selecta,c,avg(y),min(z)groupbya,cwhere;遍历中的第三种分组
  能一次返回多个结果集,那就可以大幅提高性能了。
  可惜,SQL没有这种语法,写不出这样的语句,只能用个变通的办法,就是用groupa,b,c,d的写法先算出更细致的分组结果集,但要先存成一个临时表,才能进一步用SQL计算出目标结果。SQL大致如下:createtableTtempasselecta,b,c,d,sum(casewhenthenxelse0end)sumx,max(casewhenthenyelsenullend)maxy,sum(casewhenthenyelse0end)sumy,count(casewhenthen1elsenullend)county,min(casewhenthenzelsenullend)minzgroupbya,b,c,d;selecta,b,sum(sumx)fromTtempgroupbya,bwhere;selectc,d,max(maxy)fromTtempgroupbyc,dwhere;selecta,c,sum(sumy)sum(county),min(minz)fromTtempgroupbya,cwhere;
  这样只要遍历一次了,但要把不同的WHERE条件转到前面的casewhen里,代码复杂很多,也会加大计算量。而且,计算临时表时分组字段的个数变得很多,结果集就有可能很大,最后还对这个临时表做多次遍历,计算性能也快不了。大结果集分组计算还要硬盘缓存,本身性能也很差。
  还可以用存储过程的数据库游标把数据一条一条fetch出来计算,但这要全自己实现一遍WHERE和GROUP的动作了,写起来太繁琐不说,数据库游标遍历数据的性能只会更差!
  只能干瞪眼!
  TopN运算同样会遇到这种无奈。举个例子,用Oracle的SQL写top5大致是这样的:selectfrom(selectxfromTorderbyxdesc)whererownum5
  表T有10亿条数据,从SQL语句来看,是将全部数据大排序后取出前5名,剩下的排序结果就没用了!大排序成本很高,数据量很大内存装不下,会出现多次硬盘数据倒换,计算性能会非常差!
  避免大排序并不难,在内存中保持一个5条记录的小集合,遍历数据时,将已经计算过的数据前5名保存在这个小集合中,取到的新数据如果比当前的第5名大,则插入进去并丢掉现在的第5名,如果比当前的第5名要小,则不做动作。这样做,只要对10亿条数据遍历一次即可,而且内存占用很小,运算性能会大幅提升。
  这种算法本质上是把TopN也看作与求和、计数一样的聚合运算了,只不过返回的是集合而不是单值。SQL要是能写成这样:selecttop(x,5)fromT就能避免大排序了。
  然而非常遗憾,SQL没有显式的集合数据类型,聚合函数只能返回单值,写不出这种语句!
  不过好在全集的TopN比较简单,虽然SQL写成那样,数据库却通常会在工程上做优化,采用上述方法而避免大排序。所以Oracle算那条SQL并不慢。
  但是,如果TopN的情况复杂了,用到子查询中或者和JOIN混到一起的时候,优化引擎通常就不管用了。比如要在分组后计算每组的TopN,用SQL写出来都有点困难。Oracle的SQL写出来是这样:selectfrom(selecty,x,rownumber()over(partitionbyyorderbyxdesc)rnfromT)wherern5
  这时候,数据库的优化引擎就晕了,不会再采用上面说的把TopN理解成聚合运算的办法。只能去做排序了,结果运算速度陡降!
  假如SQL的分组TopN能这样写:selecty,top(x,5)fromTgroupbyy
  把top看成和sum一样的聚合函数,这不仅更易读,而且也很容易高速运算。
  可惜,不行。
  还是干瞪眼!
  关联计算也是很常见的情况。以订单和多个表关联后做过滤计算为例,SQL大体是这个样子:selecto。oid,o。orderdate,o。amountfromordersoleftjoincityciono。cityidci。cityidleftjoinshippershono。shidsh。shidleftjoinemployeeeono。eide。eidleftjoinsuppliersuono。suidsu。suidwhereci。stateNewYorkande。titlemanagerand。。。
  订单表有几千万数据,城市、运货商、雇员、供应商等表数据量都不大。过滤条件字段可能会来自于这些表,而且是前端传参数到后台的,会动态变化。
  SQL一般采用HASHJOIN算法实现这些关联,要计算HASH值并做比较。每次只能解析一个JOIN,有N个JOIN要执行N遍动作,每次关联后都需要保持中间结果供下一轮使用,计算过程复杂,数据也会被遍历多次,计算性能不好。
  通常,这些关联的代码表都很小,可以先读入内存。如果将订单表中的各个关联字段预先做序号化处理,比如将雇员编号字段值转换为对应雇员表记录的序号。那么计算时,就可以用雇员编号字段值(也就是雇员表序号),直接取内存中雇员表对应位置的记录,性能比HASHJOIN快很多,而且只需将订单表遍历一次即可,速度提升会非常明显!
  也就是能把SQL写成下面的样子:selecto。oid,o。orderdate,o。amountfromordersoleftjoincitycono。cidc。订单表的城市编号通过序号关联城市表leftjoinshippershono。shidsh。订单表运货商号通过序号关联运货商表leftjoinemployeeeono。eide。订单表的雇员编号通过序号关联雇员表leftjoinsuppliersuono。suidsu。订单表供应商号通过序号关联供应商表whereci。stateNewYorkande。titlemanagerand。。。
  可惜的是,SQL使用了无序集合概念,即使这些编号已经序号化了,数据库也无法利用这个特点,不能在对应的关联表这些无序集合上使用序号快速定位的机制,只能使用索引查找,而且数据库并不知道编号被序号化了,仍然会去计算HASH值和比对,性能还是很差!
  有好办法也实施不了,只能再次干瞪眼!
  还有高并发帐户查询,这个运算倒是很简单:selectid,amt,tdate,fromTwhereid10100andtdatetodate(20210110,yyyyMMdd)andtdatetodate(20210125,yyyyMMdd)and
  在T表的几亿条历史数据中,快速找到某个帐户的几条到几千条明细,SQL写出来并不复杂,难点是大并发时响应速度要达到秒级甚至更快。为了提高查询响应速度,一般都会对T表的id字段建索引:createindexindexT1onT(id)
  在数据库中,用索引查找单个帐户的速度很快,但并发很多时就会明显变慢。原因还是上面提到的SQL无序理论基础,总数据量很大,无法全读入内存,而数据库不能保证同一帐户的数据在物理上是连续存放的。硬盘有最小读取单位,在读不连续数据时,会取出很多无关内容,查询就会变慢。高并发访问的每个查询都慢一点,总体性能就会很差了。在非常重视体验的当下,谁敢让用户等待十秒以上?!
  容易想到的办法是,把几亿数据预先按照帐户排序,保证同一帐户的数据连续存储,查询时从硬盘上读出的数据块几乎都是目标值,性能就会得到大幅提升。
  但是,采用SQL体系的关系数据库并没有这个意识,不会强制保证数据存储的物理次序!这个问题不是SQL语法造成的,但也和SQL的理论基础相关,在关系数据库中还是没法实现这些算法。
  那咋办?只能干瞪眼吗?
  不能再用SQL和关系数据库了,要使用别的计算引擎。
  开源的集算器SPL基于创新的理论基础,支持更多的数据类型和运算,能够描述上述场景中的新算法。用简单便捷的SPL写代码,在短时间内能大幅提高计算性能!
  上面这些问题用SPL写出来的代码样例如下:
  一次遍历计算多种分组
  A
  B
  1hrA1file(T。ctx)。open()。cursor(a,b,c,d,x,y,z)
  2hrcursorA1
  A2。select()。groups(a,b;sum(x))
  3hr定义遍历中的第一种过滤、分组
  4hrcursor
  A4。select()。groups(c,d;max(y))
  5hr定义遍历中的第二种过滤、分组
  6hrcursor
  A6。select()。groupx(a,c;avg(y),min(z))
  7hr定义遍历中的第三种过滤、分组
  8hr
  定义结束,开始计算三种方式的过滤、分组
  用聚合的方式计算Top5
  全集Top5(多线程并行计算)
  A
  1hrfile(T。ctx)。open()
  2hrA1。cursorm(x)。total(top(5,x),top(5,x))
  3hrtop(5,x)计算出x最大的前5名,top(5,x)是x最小的前5名。
  分组Top5(多线程并行计算)
  A
  1hrfile(T。ctx)。open()
  2hrA1。cursorm(x,y)。groups(y;top(5,x),top(5,x))
  用序号做关联的SPL代码:
  系统初始化
  A
  2hrenv(city,file(city。btx)。importb()),env(employee,file(employee。btx)。importb()),。。。
  3hr系统初始化时,几个小表读入内存
  查询
  A
  1hrfile(orders。ctx)。open()。cursor(cid,eid,)。switch(cid,city:;eid,employee:;)
  2hrA1。select(cid。stateNewYorkeid。titlemanager)
  3hr先序号关联,再引用关联表字段写过滤条件
  高并发帐户查询的SPL代码:
  数据预处理,有序存储
  A
  B
  1hrfile(Toriginal。ctx)。open()。cursor(id,tdate,amt,)
  2hrA1。sortx(id)
  file(T。ctx)
  3hrB2。creater(id,tdate,amt,)。appendi(A2)
  4hrB2。open()。index(indexid;id)
  5hr将原数据排序后,另存为新表,并为帐号建立索引
  帐户查询
  A
  B
  1hrT。icursor(;id10100tdatedate(20210110)tdatedate(20210125),indexid)。fetch()
  2hr查询代码非常简单
  除了这些简单例子,SPL还能实现更多高性能算法,比如有序归并实现订单和明细之间的关联、预关联技术实现多维分析中的多层维表关联、位存储技术实现上千个标签统计、布尔集合技术实现多个枚举值过滤条件的查询提速、时序分组技术实现复杂的漏斗分析等等。
  正在为SQL性能优化头疼的小伙伴们,来和我们一起探讨吧:
  《慢得受不了的查询跑批》
  识别二维码打开该页面
  重磅!开源SPL交流群成立了
  简单好用的SPL开源啦!
  为了给感兴趣的小伙伴们提供一个相互交流的平台,
  特地开通了交流群(群完全免费,不广告不卖课)
  需要进群的朋友,可长按扫描下方二维码

春日出街上班穿衣搭配春季长薄款风衣相对于短款来说更为优雅,给人温暖又帅酷的风衣,随便搭配就能流露出女人魅力,时尚经典的POLO领,凸显气质女王,内搭针织衫与修身裤,白领时尚气质油然而生。舒适……彻底告别刘海曝iPhone15全系叹号屏根据现有爆料,今年的iPhone14系列将出现刘海屏和全新叹号屏(双挖孔)共存的局面。不过,一份路线图显示,明年的iPhone15系列预计将全系换用叹号屏设计语言。……持续压制柯洁!多项第一,上半年奖金高达7亿,申真谞如此称雄韩第一人申真谞,为什么自从登上世界冠军宝座后,便频频对中国大将持续压制,尤其是在夺得去年农心杯后,申真谞对中国大将的胜率便一升再升,这也导致了柯洁的祸言,让韩国棋院提出柯申十番……7月7日小暑至!牢记喝2汤吃4菜食2果,整个伏天没烦恼小暑时节,风中都夹杂着一丝热浪,由于温度太高,就连蟋蟀都不得不离开田野,悄悄躲到庭院的墙角下避暑。这时候,人体自然也是抵挡不住如同桑拿房一般的大地,尤其是平日里工作量大的人,更……关晓彤想要幸运,只有努力,想要走得更远,就是不断向前走关晓彤一个不断蜕变的女孩。从青涩变优雅,这离不开她不断的努力,她一直给人的感觉很乖,所以大家都叫她国民闺女。关晓彤是学霸,不单单只是颜值高,这样的女孩,不一样。关晓彤对于……美国一公司推出新型电池成本不到锂电池一半电动车价格会大降?动力电池的成本,最高时占据纯电动汽车总成本的40,所以电池的价格决定着纯电车售价。2021年锂电池价格大涨,导致纯电车的价格水涨船高,阻碍了纯电车的普及速度,如果来降低电……被欺负够了的国服玩家,是时候对双标开启维权保卫战了国内的游戏市场,一度是代理游戏的天下。对于很多有足够游戏年龄的老玩家而言,自己最先接触的游戏差不多是舶来品,因为天然的对那些外来游戏的憧憬感,让很多玩家对国产也充满了轻慢的态度……转瞬如流星,看FENDI以碎片书写永恒过去、现在、和未来,将在记忆的回环中彼此实现。FENDI2022年秋冬高级定制系列于2022年7月7日发布。KimJones表示:本季,我想跳出罗马,或者至少将其置……动力电池回收行业研究从0到1000,尽享行业发展红利(报告出品方作者:天风证券,郭丽丽)1。动力电池回收:汽车电动化浪潮下的长景气赛道1。1。前言:动力电池装机放量,回收再利用市场潜力十足受新能源车销量走高的推动,我……从200万到40亿,一个街头混混如何逆袭成MMA皇帝?在商业奇才白大拿的掌舵下,UFC正大踏步向前迈。作者:果子离来源:快刀财经(ID:kuaidaocaijing)2017年8月26日,一场总价值高达6亿美金的……荟民鲜生无人零售成为行业新风口后疫情时代,无人经济的场景需求不断被扩大,无接触配送从‘应急态’趋势逐渐变为‘新常态’。无人零售逐渐成为行业新风口,荟民鲜生应运而生。荟民鲜生社区智能生鲜柜24小时无人自……蓝厂来了一波小高潮后,绿厂也按耐不住了,多款机型即将发布整个四月五原本以为是vivo的天下,毕竟vivo在这个月发布了多款机型,有折叠屏vivoXfold,有高端旗舰vivonoteX,还有iQOONEO6,另外还有一款vivo的平……
尤伯杯陈雨菲开门红,何冰娇三局险胜,中国队横扫印尼进四强北京时间5月12日,2022年汤尤杯总决赛在泰国曼谷继续进行,中国队在尤伯杯闯入四强。在与印尼队的四分之一决赛里,陈雨菲为球队打响头炮,陈清晨贾一凡和何冰娇之后各胜一场,中国队……内河运输开启纯电动时代!江苏这样打开船舶产业发展新空间全国首艘120标箱纯电动内河集装箱船江远百合号不久前在太仓投用以来,一直往返于太仓港和京杭运河苏州工业园区港间。作为航运新历史的书写者之一,船长何广科兴奋地表示,电动船零排放、……罗莱生活前三季度净利3。92亿元,投资收益同比增约135新京报贝壳财经讯(记者王真真)10月25日,罗莱生活科技股份有限公司(以下简称罗莱生活)发布2022年第三季度报告。第三季度,罗莱生活实现营业收入13。36亿元,同比减少9。2……片仔癀从一药难求到卖不出去,都是炒作惹的祸小药丸也有大力量,医药产品属特定商品种类,除特殊情况下,一般价格波动不大。但片仔癀的价格却在近两年里经历了大起与大落。去年片仔癀的价格突飞猛涨,单粒价格达到了千元以上,并……巨头宣布完全退出!解雇所有员工据央视财经援引路透社28日报道,美国戴尔公司宣布完全退出俄罗斯市场。公司发言人称,戴尔于8月中旬关闭了所有俄罗斯办事处,并停止了在俄罗斯的所有业务。据央视财经援引俄科技媒……凌晨2点!西班牙人突曝意外决定,中国球迷吐槽武磊首发彻底没戏北京时间5月16日凌晨2点,欧战联赛传来最新消息,据西班牙媒体《LaGrada》报道,西班牙人正在寻找新任主帅,目前迭戈马丁内斯是首选目标,在与卡纳瓦罗传出绯闻的情况下,西班牙……卡普空偷懒了?终于等到生化8第三人称模式,伊森还是无脸人发布于2017年的生存恐怖游戏《生化危机7》,发布那天起就颠覆了整个系列的玩法。游戏中所对抗的敌人不仅从丧尸变为了真菌,游戏视角也从前作中的第三人称改为了第一人称。将视角……谁赚钱谁又亏钱?两网电力八大金刚43家公司二季报,谁领风骚?伴随着两网电力八大金刚43家A股上市公司2022中报出炉,其二季报也同时浮出水面。作为衡量季度运营的一个重要指标,43家公司的运营情况怎么样?相对于2022一季度,……又上九嵕山作者:杜晓辉,字建辉,陕西武功人,中国作家协会会员、中国散文学会会员。昨天,我们在昭陵博物馆参加完毕张艾的新书分享会后,在礼泉文友的邀请与陪同下,又一次去登九嵕山,上昭陵……曾经那么可爱的5个童星,如今长大后,长相差距太大了事实上,对于那些童星来说,当他们年轻的时候,他们真的很可爱,深受人们的喜爱。然而,当他们长大后,往往不尽如人意。有些人们看见他们完全看不出曾经那可爱的样子,长相也被人吐槽不已。……从3199元跌至2299元,12GB256GBIMX766,大家好,我是唐三,随着近年来手机数码行业的高速发展,市面上手机种类繁多,有主打影像的旗舰手机,也有主打性能的游戏竞技类手机,但每个年龄段对手机的需求都不尽相同,学生群体更喜欢玩……7岁夏洛特与乔治聊天,被卡米拉严厉训斥,她后面的梅根眼神凌厉昨天,夏洛特公主参加了女王的葬礼,这应该算是她人生中的一个重大事件了。当天这位小公主是遇事不惊,时刻遵守着王室礼仪,有时候甚至比大两岁的哥哥乔治王子还要沉稳大气,因此她获得了人……
友情链接:易事利快生活快传网聚热点七猫云快好知快百科中准网快好找文好找中准网快软网