成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

自己動(dòng)手寫SQL執(zhí)行引擎

運(yùn)維 數(shù)據(jù)庫(kù)運(yùn)維
在造輪子的過程中一開始是非常有激情非常快樂的。但隨著系統(tǒng)越來越龐大,復(fù)雜性越來越高,進(jìn)度就會(huì)越來越慢,還時(shí)不時(shí)要推翻自己原來的設(shè)想并重新設(shè)計(jì),然后再協(xié)同修改關(guān)聯(lián)的所有代碼,就如同泥沼,越陷越深。

[[344553]]

自己動(dòng)手寫SQL執(zhí)行引擎

前言

在閱讀了大量關(guān)于數(shù)據(jù)庫(kù)的資料后,筆者情不自禁產(chǎn)生了一個(gè)造數(shù)據(jù)庫(kù)輪子的想法。來驗(yàn)證一下自己對(duì)于數(shù)據(jù)庫(kù)底層原理的掌握是否牢靠。在筆者的github中給這個(gè)database起名為Freedom。

整體結(jié)構(gòu)

既然造輪子,那當(dāng)然得從前端的網(wǎng)絡(luò)協(xié)議交互到后端的文件存儲(chǔ)全部給擼一遍。下面是Freedom實(shí)現(xiàn)的整體結(jié)構(gòu),里面包含了實(shí)現(xiàn)的大致模塊:

 

最終存儲(chǔ)結(jié)構(gòu)當(dāng)然是使用經(jīng)典的B+樹結(jié)構(gòu)。當(dāng)然在B+樹和文件系統(tǒng)block塊之間的轉(zhuǎn)換則通過Buffer(Page) Manager來進(jìn)行。當(dāng)然了,為了完成事務(wù),還必須要用WAL協(xié)議,其通過Log Manager來操作。

Freedom采用的是索引組織表,通過DruidSQL Parse來將sql翻譯為對(duì)應(yīng)的索引操作符進(jìn)而進(jìn)行對(duì)應(yīng)的語(yǔ)義操作。

MySQL Protocol結(jié)構(gòu)

client/server之間的交互采用的是MySQL協(xié)議,這樣很容易就可以和mysql client以及jdbc進(jìn)行交互了。

query packet

mysql通過3byte的定長(zhǎng)包頭去進(jìn)行分包,進(jìn)而解決tcp流的讀取問題。再通過一個(gè)sequenceId來在應(yīng)用層判斷packet是否連續(xù)。

 

result set packet

mysql協(xié)議部分最復(fù)雜的內(nèi)容是其對(duì)于result set的讀取,在NIO的方式下加重了復(fù)雜性。

Freedom通過設(shè)置一系列的讀取狀態(tài)可以比較好的在Netty框架下解決這一問題。

 

row packet

還有一個(gè)較簡(jiǎn)單的是對(duì)row格式進(jìn)行讀取,如上圖所示,只需要按部就班的解析即可。

 

由于協(xié)議解析部分較為簡(jiǎn)單,在這里就不再贅述。

SQL Parse

Freedom采用成熟好用的Druid SQL Parse作為解析器。事實(shí)上,解析sql就是將用文本表示

的sql語(yǔ)義表示為一系列操作符(這里限于篇幅原因,僅僅給出select中where過濾的原理)。

對(duì)where的處理

例如where后面的謂詞就可以表示為一系列的以樹狀結(jié)構(gòu)組織的SQL表達(dá)式,如下圖所示:

當(dāng)access層通過游標(biāo)提供一系列row后,就可以通過這個(gè)樹狀表達(dá)式來過濾出符合where要求的數(shù)據(jù)。Druid采用了Parse中常用的visitor很方便的處理上面的表達(dá)式計(jì)算操作。

對(duì)join的處理

對(duì)join最簡(jiǎn)單處理方案就是對(duì)兩張表進(jìn)行笛卡爾積,然后通過上面的where condition進(jìn)行過濾,如下圖所示:

 

Freedom對(duì)于縮小笛卡爾積的處理

由于Freedom采用的是B+樹作為底層存儲(chǔ)結(jié)構(gòu),所以可以通過where謂詞來界定B+樹scan(搜索)的范圍(也即最大搜索key和最小搜索key在B+樹種中的位置)。考慮sql

  1. select a.*,b.* from t_archer as a join t_rider as b where a.id>=3 and a.id<=11 b.id and b.id>=19 b.id<=31 

那么就可以界定出在id這個(gè)索引上,a的scan范圍為[3,11],如下圖所示:

 

b的scan范圍為[19,31],如下圖所示(假設(shè)兩張表數(shù)據(jù)一樣,便于繪圖):

 

scan少了從原來的15*15(一共15個(gè)元素)次循環(huán)減少到4*4次循環(huán),即循環(huán)次數(shù)減少到7.1%

當(dāng)然如果存在join condition的話,那么Freedom在底層cursor遞歸處理的過程中會(huì)預(yù)先過濾掉一部分?jǐn)?shù)據(jù),進(jìn)一步減少上層的過濾。

B+Tree的磁盤結(jié)構(gòu)

leaf磁盤結(jié)構(gòu)

Freedom的B+Tree是存儲(chǔ)到磁盤里的。考慮到存儲(chǔ)的限制以及不定長(zhǎng)的key值,所以會(huì)變得非常復(fù)雜。Freedom以page為單位來和磁盤進(jìn)行交互。葉子節(jié)點(diǎn)和非葉子節(jié)點(diǎn)都由page承載并刷入磁盤。結(jié)構(gòu)如下所示:

 

一個(gè)元組(tuple/item)在一個(gè)page中分為定長(zhǎng)的ItemPointer和不定長(zhǎng)的Item兩部分。

其中ItemPointer里面存儲(chǔ)了對(duì)應(yīng)item的起始偏移和長(zhǎng)度。同時(shí)ItemPointer和Item如圖所示是向著中心方向進(jìn)行伸張,這種結(jié)構(gòu)很有效的組織了非定長(zhǎng)Item。

leaf和node節(jié)點(diǎn)在Page中的不同

雖然leaf和node在page中組織結(jié)構(gòu)一致,但其item包含的項(xiàng)確有區(qū)別。由于Freedom采用的是索引組織表,所以對(duì)于leaf在聚簇索引(clusterIndex)和二級(jí)索引(secondaryIndex)中對(duì)item的表示也有區(qū)別,如下圖所示:

 

其中在二級(jí)索引搜索時(shí)通過secondaryIndex通過index-key找到對(duì)應(yīng)的clusterId,再通過

clusterId在clusterIndex中找到對(duì)應(yīng)的row記錄。

由于要落盤,所以Freedom在node節(jié)點(diǎn)中的item里面寫入了index-key對(duì)應(yīng)的pageno,

這樣就可以容易的從磁盤恢復(fù)所有的索引結(jié)構(gòu)了。

B+Tree在文件中的組織

有了Page結(jié)構(gòu),我們就可以將數(shù)據(jù)承載在一個(gè)個(gè)page大小的內(nèi)存里面,同時(shí)還可以將page刷新到對(duì)應(yīng)的文件里。有了node.item中的pageno,我們就可以較容易的進(jìn)行文件和內(nèi)存結(jié)構(gòu)之間的互相映射了。

B+樹在磁盤文件中的組織如下圖所示:

 

B+樹在內(nèi)存中相對(duì)應(yīng)的映射結(jié)構(gòu)如下圖所示:

 

文件page和內(nèi)存page中的內(nèi)容基本是一致的,除了一些內(nèi)存page中特有的字段,例如dirty等。

每個(gè)索引一個(gè)B+樹

在Freedom中,每個(gè)索引都是一顆B+樹,對(duì)記錄的插入和修改都要對(duì)所有的B+樹進(jìn)行操作。

B+Tree的測(cè)試

筆者通過一系列測(cè)試case,例如隨機(jī)變長(zhǎng)記錄對(duì)B+樹進(jìn)行插入并落盤,修復(fù)了其中若干個(gè)非常詭異的corner case。

B+Tree的todo

筆者這里只是完成了最簡(jiǎn)單的B+樹結(jié)構(gòu),沒有給其添加并發(fā)修改的鎖機(jī)制,也沒有在B+樹做操作的時(shí)候記錄log來保證B+樹在宕機(jī)等災(zāi)難性情況下的一致性,所以就算完成了這么多的工作量,距離一個(gè)高并發(fā)高可用的bptree還有非常大的距離。

Meta Data

table的元信息由create table所創(chuàng)建。創(chuàng)建之后會(huì)將元信息落盤,以便Freedom在重啟的時(shí)候加載表信息。每張表的元信息只占用一頁(yè)的空間,依舊復(fù)用page結(jié)構(gòu),主要保存的是聚簇索引和二級(jí)索引的信息。元信息對(duì)應(yīng)的Item如下圖所示:

 

如果想讓mybatis可以自動(dòng)生成關(guān)于Freedom的代碼,還需實(shí)現(xiàn)一些特定的sql來展現(xiàn)Freedom的元信息。這個(gè)在筆者另一個(gè)項(xiàng)目rider中有這樣的實(shí)現(xiàn)。原理如下圖所示:

 

實(shí)現(xiàn)了上述4類SQL之后,mybatis-generator就可以通過jdbc從Freedom獲取元信息進(jìn)而自動(dòng)生成代碼了。

事務(wù)支持

由于當(dāng)前Freedom并沒有保證并發(fā),所以對(duì)于事務(wù)的支持只做了最簡(jiǎn)單的WAL協(xié)議。通過記錄redo/undolog從而實(shí)現(xiàn)原子性。

redo/undo log協(xié)議格式

Freedom在每做一個(gè)修改操作時(shí),都會(huì)生成一條日志,其中記錄了修改前(undo)和修改后(redo)的行信息,redo用來回滾,redo用來宕機(jī)recover。結(jié)構(gòu)如下圖所示:

 

WAL協(xié)議

WAL協(xié)議很好理解,就是在事務(wù)commit前將當(dāng)前事務(wù)中所產(chǎn)生的的所有l(wèi)og記錄刷入磁盤。

Freedom自然也做了這個(gè)操作,使得可以在宕機(jī)后通過log恢復(fù)出所有的數(shù)據(jù)。

 

回滾的實(shí)現(xiàn)

由于日志中記錄了undo,所以對(duì)于一個(gè)事務(wù)的回滾直接通過日志進(jìn)行undo即可。如下圖所示:

 

宕機(jī)恢復(fù)

Freedom如果在page全部刷盤之后關(guān)機(jī),則可以由通過加載page的方式獲取原來的數(shù)據(jù)。

但如果突然宕機(jī),例如kill -9之后,則可以通過WAL協(xié)議中記錄的redo/undo log來重新

恢復(fù)所有的數(shù)據(jù)。由于時(shí)間和精力所限,筆者并沒有實(shí)現(xiàn)基于LSN的檢查點(diǎn)機(jī)制。

Freedom運(yùn)行

  1. git clone https://github.com/alchemystar/Freedom.git 
  2. // 并沒有做打包部署的工作,所以最簡(jiǎn)單的方法是在java編輯器里面 
  3. run alchemystar.freedom.engine.server.main 

以下是筆者實(shí)際運(yùn)行Freedom的例子:


 

 

join查詢

 

delete回滾

 

Freedom todo

Freedom還有很多工作沒有完成,例如有層次的鎖機(jī)制和MVCC等,由于工作忙起來就耽擱了。

于是筆者就看了看MySQL源碼的實(shí)現(xiàn)理解了一下鎖和MVCC實(shí)現(xiàn)原理,并寫了兩篇博客。比起

自己動(dòng)手?jǐn)]實(shí)在是輕松太多了^_^。

MVCC

https://my.oschina.net/alchemystar/blog/1927425

二階段鎖

https://my.oschina.net/alchemystar/blog/1438839

尾聲

在造輪子的過程中一開始是非常有激情非常快樂的。但隨著系統(tǒng)越來越龐大,復(fù)雜性越來越高,進(jìn)度就會(huì)越來越慢,還時(shí)不時(shí)要推翻自己原來的設(shè)想并重新設(shè)計(jì),然后再協(xié)同修改關(guān)聯(lián)的所有代碼,就如同泥沼,越陷越深。至此,筆者才領(lǐng)悟了軟件工程最重要的其實(shí)是控制復(fù)雜度!始終保持簡(jiǎn)潔的接口和優(yōu)雅的設(shè)計(jì)是實(shí)現(xiàn)一個(gè)大型系統(tǒng)的必要條件。

本文轉(zhuǎn)載自微信公眾號(hào)「解Bug之路」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系解Bug之路公眾號(hào)。

 

責(zé)任編輯:武曉燕 來源: 解Bug之路
相關(guān)推薦

2014-11-26 10:54:20

C#

2015-07-23 14:53:50

貝葉斯分類器

2017-03-02 13:31:02

監(jiān)控系統(tǒng)

2018-09-12 10:58:11

NBA數(shù)據(jù)存儲(chǔ)

2018-02-07 10:46:20

數(shù)據(jù)存儲(chǔ)

2015-06-02 10:24:43

iOS網(wǎng)絡(luò)請(qǐng)求降低耦合

2015-06-02 09:51:40

iOS網(wǎng)絡(luò)請(qǐng)求封裝接口

2023-12-15 10:14:42

數(shù)據(jù)庫(kù)select語(yǔ)句

2023-12-16 13:21:00

Python元類ORM

2011-08-25 09:30:22

2020-05-20 13:53:41

HTTP環(huán)境安裝

2015-09-01 09:49:28

2015-06-02 09:41:00

iOS網(wǎng)絡(luò)請(qǐng)求NSURLSessio

2011-07-19 09:36:24

SplitSQL

2014-06-20 09:18:54

Dustjs中間件

2009-03-16 16:30:18

2014-03-17 10:34:48

SQL Server

2009-10-27 09:01:54

VB.NET內(nèi)存指針

2011-05-06 15:35:58

打印機(jī)打印故障

2009-12-03 13:56:05

Suse Linux開xinetd
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 欧美日韩精品久久久免费观看 | 国产一区二区视频在线 | 在线国产视频 | 99国产精品99久久久久久 | 亚洲高清三级 | 91精品中文字幕一区二区三区 | 亚洲草草视频 | 国产一级视频在线播放 | gogo肉体亚洲高清在线视 | 在线中文字幕av | 亚洲午夜一区二区 | 亚洲精品永久免费 | 欧美日本一区二区 | 中文字幕亚洲精品在线观看 | 天天操天天摸天天爽 | 99国产精品99久久久久久粉嫩 | 午夜在线| 亚洲精品视频一区二区三区 | 激情婷婷 | 在线一区视频 | 777zyz色资源站在线观看 | 精品久久99 | 日韩精品一区在线 | 99精品免费久久久久久日本 | 久久综合久久综合久久 | 欧美激情精品久久久久久变态 | 国产日韩欧美一区二区 | 亚洲激情在线 | 9999精品视频| 亚洲欧美日韩一区二区 | 国产一区二区三区色淫影院 | 麻豆一区二区三区精品视频 | 亚洲精品视频在线观看视频 | 中文字幕一区二区三区不卡 | 久久成人精品视频 | 成人亚洲在线 | 国产精品视频www | 亚洲一区综合 | 2019天天干夜夜操 | 国产高清精品网站 | 亚洲日日夜夜 |