一、为什么学习数据库?

学习数据库,你肯定需要先了解它,这样你才知道你为什么要学习它,数据库除了SQL还有一些别的数据库,了解它们的作用,可以在不同的场景选择最符合的一个数据库。

1.1 什么是数据库?

首先你的先知道数据库的作用是什么,数据库的定义:数据库是结构化信息或数据的有组织的集合,通常以电子方式存储在计算机系统中。数据库通常由数据库管理系统 (DBMS)控制。数据和 DBMS 以及与之关联的应用程序一起被称为数据库系统,通常简称为数据库。

数据库的组件:硬件,软件,数据,程序,访问语言。

1.2 数据库有啥优点?

列举以下几点,来说明为什么学习数据库:

  1. 数据库可以有效地存储大量记录(占用的空间很小)
  2. 海量的数据中很容易找到需要的信息,给你一万条数据你用眼睛找?
  3. 容易添加新的数据以及编辑或删除已有的数据
  4. 可以轻松搜索数据,比如有一对蔬菜的数据,你只想查找所有白菜的数据?
  5. 数据可以轻松排序,比如有一万个人注册网站,你想根据注册时间排序?也许只需要一行语句
  6. 多人可以同时访问同一个数据库,这样可以多人管理,而且每个人的全选可能不一样,一个大型数据库可以每个人负责一部分
  7. 安全性可能比纸质文件更好,比存储到本地的文档好,这是毋庸置疑的!

1.3 数据库有啥用途?偏偏要学它

这里我依然举例子来说明它的用途:

  1. 警方在数据库中有所有已知罪犯的详细信息,例如他们犯下的罪行
  2. 学校使用数​​据库来存储学生的详细信息,例如每个人的成绩
  3. 医院会将所有患者的详细信息存储在数据库中,例如他们的病历
  4. 政府使用数据库存储人们的所得税支付记录
  5. 如果过路费不在人工,一个数据库用于跟踪上海中心所有已支付过路费的司机,如果未支付就通过就要对他罚款。再比如无人超市?

二、为什么不选择SQL?

我们需要知道以下SQL的优缺点,才能知道是否选择它。这里列举SQL的缺点:

  1. 安全:无论 SQL哪个 版本,SQL 中的数据库都经常受到威胁,因为它拥有大量敏感数据。
  2. 速度:随着 SQL 数据库变得越来越大,大量表之间所需的查找和连接可能会减慢速度!
  3. 可靠性:SQL 很难实现高性能算法,大数据操作的性能只能依靠数据库的优化引擎,但在复杂情况下往往有些不可靠。

2.1 使用SQL处理亿量数据太难了!

很多大数据计算都是用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 rownum<=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 (select y,x,row\_number() over (partition by y order by x desc) rn from T) where rn<=5

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

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

 select y,top(x,5) from T group by y

把top看成和sum一样的聚合函数,这不仅更易读,而且也很容易高速运算。可惜,不行。还是干瞪眼!只想说一句:气死

2.2 关联计算

关联计算也是很常见的情况。以订单和多个表关联后做过滤计算为例,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.shid=sh.shid left join employee e on o.eid=e.eid left join supplier su on o.suid=su.suid
where ci.state='New York' and e.title='manager' 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.shid=sh.# --订单表运货商号通过序号#关联运货商表 left join employee e on o.eid=e.# --订单表的雇员编号通过序号#关联雇员表 left join supplier su on o.suid=su.#--订单表供应商号通过序号#关联供应商表
where ci.state='New York' and e.title='manager' and ...

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

2.3 高并发帐户查询

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

select id,amt,tdate,… from T
where id='10100' and tdate>= to\_date('2021-01-10','yyyy-MM-dd') and tdate<to_date('2021-01-25','yyyy-mm-dd') and="" …="" <p="">

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

create index index_T_1 on T(id)

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

三、为什么选择SPL?

不能再用SQL和关系数据库了,要使用别的计算引擎,那就是SPL。开源的集算器SPL基于创新的理论基础,支持更多的数据类型和运算,能够描述上述场景中的新算法。用简单便捷的SPL写代码,在短时间内能大幅提高计算性能!very good!

3.1 解决SQL的痛!

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

  • 一次遍历计算多种分组
A B
1 =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
1 >env(city,file(“city.btx”).import@b()),env(employee,file(“employee.btx”).import@b()),…
2 //系统初始化时,几个小表读入内存

查询

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
1 =T.icursor(;id==10100 && tdate>=date(“2021-01-10”) && tdate<date(“2021-01-25”) && …,index_id).fetch()
2 //查询代码非常简单

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

3.2 更多关于SPL的思路和算法

这里还有更多SPL代码以体现其思路及大数据算法:

  • 性能优化技巧:遍历复用提速多次分组
  • 性能优化技巧:TopN
  • 性能优化技巧:预关联
  • 性能优化技巧:部分预关联
  • 性能优化技巧:外键序号化
  • 性能优化技巧:维表过滤或计算时的关联
  • 性能优化技巧:有序归并
  • 性能优化技巧:有序定位关联提速主子关联后的过滤
  • 性能优化技巧:小事实表与大维表关联

正在为SQL性能优化头疼的小伙伴们,可以来一起讨论:http://www.raqsoft.com.cn/wx/Query-run-batch-ad.html

四、SPL参考

  • SPL官网
  • SPL下载地址
  • github开源SPL地址

一起支持国产数据库,点个star支持一下吧:

只知道SQL数据库?又一国产数据库语言诞生了相关推荐

  1. oracle数据库内容替换,国产数据库能否替换oracle数据库?

    国产数据库能不能替换oracle?能,但是困难一定很多.没有什么数据库是无法替代的,只是替代的代价到底有多高.迁移替代需要考虑很多方面: 1 性能 首先是性能有没有oracle好.其实是大部分国产数据 ...

  2. jpa 人大金仓数据库方言_国产数据库的春天,人大金仓完成近亿元融资

    近日,人大金仓宣布完成近亿元融资.本次融资原股东太极股份等追加增资,同时引入战略投资方:电科研投.南威软件与东华软件,总融资额近亿元. 持续加码产品研发与市场拓展 本次融资将助力人大金仓持续加码产品研 ...

  3. 国产数据库替换oracle,国产数据库能否替换oracle数据库?

    对于Oracle能否被开源系衍生的数据库代替这个问题,个人觉得需要从两个维度去看. 第一个维度,技术维度.从技术维度来展开的话,首先需要看的就是产品的成熟度,这一点来讲,Oracle从三足鼎立的时代发 ...

  4. 2020 数据技术嘉年华:吹响国产数据库的集结号 诚邀参会览技术前沿

    汇聚数据精英,引领行业发展,数据技术嘉年华已经走入了第十个年头.第十届数据技术嘉年华(DTC 2020)将于2020年11月20-21日在北京丽都皇冠假日酒店盛大开启. 大会以"自研·智能· ...

  5. 武林大会之国产数据库风云榜-2021年11月

    神秘能量篇: 话说在盘古开天之后,有一种神秘的能量出现于天地,后人称之为"数据",人们先后利用骨制品.绳结.纸张.算盘等工具尝试存储并使用这股神秘的力量,又在计算机发明之后,信息可 ...

  6. 国产数据库之-GreatSQL

    这次的EW争端,可能让很多人重新认识了软件国产化的重要性和急迫性.对企业级应用来说,芯片.操作系统.中间件.数据库这些都是俗称"卡脖子"的技术,例如数据库,各种国产数据库,可以说是 ...

  7. 国产数据库VastBase适配指南

    背景   目前数据库市场上,仍然是甲骨文.IBM为代表的国外数据库软件处于主导地位,国产化的数据库的使用率,与推广面很有限.相对于主流数据库而言,国产数据库的优势并不明显,还有相当远的距离.那么我们为 ...

  8. 苦尽甘来之国产数据库风云榜-2022年03月

    数据库分类 随着数据库的发展,出现了很多新的词汇,例如HATP,NewSQL,分布式,云原生等,通过下面对数据库分类的介绍来了解这些概念. 关系型数据库(SQL) 关系型数据库指采用了关系模型来组织数 ...

  9. 2022年11月国产数据库大事记-墨天轮

    本文为墨天轮社区整理的2022年11月国产数据库大事件和重要产品发布消息. 文章目录 11月国产数据库大事记(时间线) 产品/版本发布 兼容认证 排行榜新增数据库 11月国产数据库大事记(时间线) 1 ...

  10. 疫情下开源数据库逆势增长,新基建下国产数据库迎机遇

    2020年5月DB-Engines 数据库流行度排行大家都看了吗? 虽然 Top 10 与上月没有任何变化,但仔细观察本月的排行榜,Oracle 较上月几乎持平,仅微涨 0.02 分:相较而言,MyS ...

最新文章

  1. 2招按摩轻松解除黑眼圈 - 生活至上,美容至尚!
  2. No module named Crypto--转
  3. php dropdownlist,遇到dropdownlist
  4. 数据库事务的四大特性以及事务的隔离级别详解
  5. 完美国际真数苹果_章子怡玩出新花样,雷人造型别有韵味!和小苹果同框犹如亲姐妹...
  6. 如何优雅的利用Windows服务来部署ASP.NET Core程序
  7. 极客招募令!兄弟杯区块链极客竞技大赛在上海等您来战!
  8. php 逻辑与运算符使用说明
  9. matlab cgf sc 未定义,matlab通用神經網絡代碼
  10. 敏捷开发免费管理工具——火星人预览之七:自定义字段
  11. 技术人生:立志、勤学、改过、责善
  12. java 字符流异常处理_IO字符流9jdk7和jdk9中流的异常的处理
  13. 使用RN开发App,引入图标失效问题的解决
  14. 汇编中的int 10h 与int 19h
  15. 08-【go】go语言中的*和的使用方法
  16. 给俺的女神翠花发邮件告白,我学会了这些网络基础知识
  17. 104届广交会秋交会展位(摊位)联营参展函 (广交会展位合作)
  18. NVIDIA vulkan driver的安装和Jetson平台上vulkan sdk的制作
  19. dataframe列时间字段提取年、月、日、时、分
  20. 【论文笔记】SeqSLAM、Fast-SeqSLAM 和 Bow、Incremental Bow、DBoW2、ORB-SLAM 论文阅读笔记

热门文章

  1. 【web开发】导出到Excel表格
  2. STM32物联网项目-RS485通信(Modbus协议)
  3. egret创建EUI动画
  4. 什么是Transformer?| 小白深度学习入门
  5. 仙境幻想游戏特色系统
  6. python函数算面积_Python函数之计算规则图形的面积
  7. 谈谈技术(Technology)
  8. CC2530设置发射功率
  9. STM32烧写失败错误:Contents mismatch at: 08000000H (Flash=FFH Required=00H)
  10. 【Unity Shader】纹理实践4.0:简单尝试渐变纹理和遮罩纹理