谈谈数据库
事务与ACID其实是为了帮程序员屏蔽底层并发的细节.
ACID Vs CAP
词语 | 传统数据库 | CAP | 迷惑? |
---|---|---|---|
事务 | 一组不可分割的操作(包括ACID) | - | - |
持久化 (Durability) | 状态更改故障/重启后不受影响 | - | - |
一致性(Consistency) | 数据本身的完整性约束(数据类型,范围.etc) | “原子一致性”模型的简称 | 不同义 |
隔离性(Isolation) | 多个事务并发执行结果可以像事务顺序执行一般 | 被包含着CAP的一致性模型中 | 被包含 |
原子性(Atomicity) | 所有更改要么全部发生,要么全部不发生 | 原子一致性模型 | 不同义 |
可用性(Availability) | 按时间计使用率 | 非故障节点应至少响应一次 | 不同义 |
分区容错性(Partition-toleranceh) | 主备数据不一致? | 两组节点分区时,它们之间消息丢失 | 不同义 |
note:
- 顺序一致性:所有处理器的内存访问是交叉的,然而程序的行为好像它们是按顺序执行的
- 原子/线性一致性:顺序一致性+实时约束.线性一致性假设所有处理器共享全局时间
- Isolation
serializable
可序列化:读写均锁表,强行单线程操作…开发人员可以躺平了repeatable read
:事务开始时建立一致性视图,读取时若DB_TRX_ID
(处理的事务数ID)在视图建立前未提交,则读到的是原来的数据(多版本并发MVCC
的基础)read committed
读取已提交数据:写的时候锁行,写完了commit再允许被读read uncommitted
读取未提交数据:什么也不做,只要内存中改了记录就能被读
- Base理论,基本可用→最终一致性
对数据库的操作依然离不开查找和排序,并且传承于通用的数据结构与算法。
2. 对传统索引的理解
2.1. 只是支持快速查找的数据结构
索引的本质在查询关键字与数据库记录之间建立的一种支持快速查找的数据结构 ,不一定是b+树,也可以是hash索引,也可以是b树/二叉树/数组/n叉树等等。
因为磁盘相比主存慢十万倍,而且磁盘一次能取一个扇区的数据1,因此索引的设计应当使每次磁盘I/O
取到内存的数据都能帮助后续记录的定位,尽可能减少IO的次数。以第二行做二叉树形式的索引结构如下:
-
B树(多路平衡二叉树)俯视图2:
-
B+树只是一种均衡各方面的产物:
操作 | Hash | 有序数组 | 二叉树 | b树(B-tree,Balanced Tree) | b+树 | b*树 |
---|---|---|---|---|---|---|
Insert | O(1) | O(n) | O(logn) | n叉树,每个节点可以有1200个叉 | n叉树,优化b树存储,提供范围查找 | n叉树,优化b+树页分裂空间利用率低的问题 |
Remove | O(logn) | 树高不超过3 | 所有Key只在叶子节点上出现一次 | 非叶子节点的兄弟节点之间也通过指针相连 | ||
Update | 最多只需要访问3次磁盘就可定位数据块 | 所有非叶子节点都是叶子节点的索引 | 如果节点上的子节点满了 | |||
Find | O(1) | O(logn) | O(logn) | 极少io—根节点总在内存中 | 叶子结点如果是聚簇索引直接定位到记录 | 就将它的子节点挪一部分到旁边没满的兄弟节点上 |
Iterator(范围查找n) | O(n) | O(n)? | 所有数据只在节点上出现一次 | 叶子结点如果是非聚簇索引存储指向记录的指针 | 避免了重新创建新的节点的过程 | |
Sort/Group by | 搜索可以在非叶子节点结束 | 各个叶子节点通过指针跟兄弟节点关联(双指针) | ||||
supplement | 子节点数目超过后再二分,称作页分裂 |
B+树正面图:
2.2. 索引的演化简史
- 索引结构经历了Hash → 二叉树 → 平衡二叉树(BST) → 多路平衡二叉树(B-Tree) → B+树 →
B*
树的演变过程,依次提供了一些功能或解决了一些问题:- 直接给每个记录唯一的键(hash值),并通过唯一键给记录定位是最直接的想法
- 不同索引键可能存在相同 Hash 值而碰撞,且不支持Key比较,且不支持范围遍历
- 支持范围查找O(log2n)
- 树结点因为包括<指针,主键,(记录))>所需空间变大
- 分叉少容易树高则查找次数变多
- 支持n叉,尽可能在3次IO内定位记录,充分利用每次磁盘I/O取到内存的索引(每一次取的数据包含下一次的索引)
- 受限于每一个数据页(结点)的存储空间
16k
(InnoDB),分开存储各级索引与记录以便每层存储更多索引,降低树深- 层数可以不变,但是能存储的更多了.假设每个结点大小为
2k
,结点中每一个指针为6-8B
- 第一层1个2k大小的结点 → 第二层200个结点 → 第三层200^2 = 40000个结点,这三层大小=2k + 0.4m + 80m,足够三层放入内存中,而第4层可以支撑200^4=16亿条数据,恰好他们也都在磁盘中.这种情况下也只有一次与磁盘的IO
- 层数可以不变,但是能存储的更多了.假设每个结点大小为
- 结点必要时换父亲,尽可能减少页分裂的次数
- 直接给每个记录唯一的键(hash值),并通过唯一键给记录定位是最直接的想法
- 红黑树虽然也是B-Tree类型的树,但不会控制树的高度,所以不适合利用磁盘存储的MySQL 的。而因为B-Tree每一块的大小是固定的,如果有新的元素处于一个满了数据块的范围内,则会触发该数据块分裂,最糟糕时情况是一次插入操作所有的数据块都分裂了一遍,则带来的问题就是存储浪费,所以它不适合内存
2.3. 索引聚不聚簇
以新华字典为例说明,新华字典提供主要的索引方法:拼音查询。因为字典本身按照这样的索引已聚集排序,当输入一个拼音如gong
—一个Key
,就可以按照g
—ong
(—笔画数
)不断缩小查找范围,最后遍历最小的那个分类集合命中汉字—一个value
.这样像拼音这样的索引被称作”聚簇索引(Clustered Index
)“。聚簇索引本身并不需要聚簇索引表。
可以看出聚簇索引并不是一种单独的索引类型,而是一种数据存储方式,满足聚簇
的字面意思3。必须注意的是,每张表有且仅有一个聚簇索引,4因为数据库只能按照一种方法进行排序。
同时字典也提供部首查询的方法,字典中字排列顺序与这里索引—部首的排序不一致,所以部首查询表本身一定需要包含命中的指向字的直接索引。有过部首查询经历会发现,部首查询表自己是按笔画数进行的划分,先按照”一笔“”二笔“”三笔“…这样类似数组下标的索引查得部首如”工”,接着同样按照笔画数如三笔—一个Key
,又查到部首索引表中的字与它对应的页数—一个<K,V>
,而这里的V
是一个指向实际物理文件—字典中字的point
.这里的部首索引表本身也就是一个含有笔画—这种聚簇索引的表,而对于字典,部首索引表的部首则是”非聚簇索引(Noinclustered Index
)“
新华字典的层级关系可以表示为
假设每页为一记录,自然地页数就是主键id,并且满足主键约束(不重复)。在页数之上添加的部首索引表便是在聚簇索引的基础上添加了非聚簇索引。 而在数据库中,索引是通过n叉树的形式进行描述的。承接上述的层级关系,聚簇索引的叶子结点是最终的数据结点记录,而非聚簇索引的叶子结点仍然是索引节点,一定有一个指向最终数据的指针。
对于使用聚簇索引方式存储的表,每个索引后都会附加索引信息?
2.4. 索引优缺点5
2.4.1. 优点
-
索引大大减少了存储引擎需要扫描的数据量。因为可以逐步聚焦需要的记录。
-
索引可以帮助存储引擎避免排序和临时表。因为对于聚簇索引本身记录会按索引组织,从而也可以避免临时表之后再建立索引。
-
索引可以将随机 I/O 变成顺序 I/O(比如B+树底部的双链表)。随机指的是无目的查询,相比有了索引关键字,只需要按步骤顺序查询或者遍历叶节点就好。
2.4.2. 缺点
-
降低更新表的速度,如对表进行 INSERT、UPDATE 和 DELETE。因为更新表时,MySQL 不仅要更新数据,还要更新索引文件。
-
索引文件会占用磁盘空间。当在一个大表上创建了多种组合索引,且伴随大量数据量插入,索引文件大小也会快速膨胀。
-
在重复列上添加聚簇索引,引擎需要自动添加额外字段以组合满足主键(
PRIMARY KEY
)UNIQUE
约束。因此对于包含重复的内容的数据列,是否建立聚簇与非聚簇索引得综合判断6。 -
- 如果是性别标签“男 女 未知”这种对应的大量重复记录,索引开销可能大于直接遍历;
- 如果是日期划分,虽然每日有许多,但是每一个按日的聚合划分仍然有效区分数据集,建立聚簇索引;
- 小数据集或数据记录本身小,直接排序更好,优先聚簇索引;
- 如果一条记录本身十分大,为了降低insert的维护开销,可用非聚簇索引(以避免对记录的重排序?)
-
对于记录很少的表,不仅仅范围查询,大部分情况下简单的全表扫描更高效。因为拿到主索引回表查一次,可能会涉及 I/O 的行数更多,同样这也是聚簇索引优于非聚簇索引,以及
select *
垃圾的原因。
Footnotes
-
唯一性并不是索引的必要条件。将索引设置为唯一,对于等值查找当查到第一条符合条件的纪录时即可停止查找,而非唯一索引则要继续查找(回表,因为叶结点还是指针)。不过索引唯一维护开销也大,每一行数据的插入都会去检查重复以保证唯一性; ↩