(本文转自微信公众号:分布式实验室,译者:吴世曦,发布于2020年3月20日)

作为Web开发工程师,我的日常工作会用到关系型数据库,但是数据库的内部工作机制对我来讲是个黑盒子,我会有这样的问题:
- 数据以怎样的格式存储在内存和硬盘上?
- 什么时候需要将数据从内存挪到硬盘上?
- 为什么每张表的主键是唯一的?
- 事务处理是如何回滚的?
- 索引的格式是怎样的?
- 全表扫描是何时,怎样发生的?
- PreparedStatement是以怎样的格式保存的?(PreparedStatement是用来执行SQL查询语句的API之一)
简而言之,数据库是如何工作的?
为了搞清这个问题,我从零写了个数据库。这个数据库是基于SQLite的,因为比起MySQL或者PostgreSQL,SQLite更加轻量级,更容易被理解。整个数据库存储在一个文件中。
SQLite
在SQLite的官网上有很多相关的文档[1],我复制了SQLite Database System:Design and Implementation[2]如下:
为了能获取或者修改数据,一条查询经过了一系列的组件链。前端的组件包括:
- tokenizer
- parser
- code generator
前端组件的输入是一条SQL的查询。而输出是SQLite的虚拟机字节码(其本质上是可以在数据库上运行的已编译程序)。
后端组件包括:
- 虚拟机
- B-tree
- pager
- os interface
虚拟机将前端生成的字节码作为指令。然后,它可以对一个或多个表或索引执行操作,每个表或索引都存储在称为B树的数据结构中。VM本质上是关于字节码指令类型的switch语句(switch语句是一种选择控制机制,用于允许变量或表达式的值通过搜索和映射来更改程序执行的控制流)。
每个B树由许多节点组成。每个节点的长度为一页。B树可以通过向pager发布指令从磁盘获取页面或者存储页面到磁盘。
Pager接收命令以读取或写入数据页。它负责以适当的偏移量在数据库文件中进行读取/写入。它还在内存中保留了最近访问页面的缓存,并确定何时需要将这些页面写回到磁盘。
os接口层会因为SQLite在哪种操作系统中编译而有所不同。本教程不支持多个平台。
千里之行始于足下,让我们直接从REPL开始。
制作一个简单的REPL
当你从命令行启动SQLite时,它会启动一个read-execute-print循环:
1~ sqlite3
2SQLite version 3.16.0 2016-11-04 19:09:39
3Enter ".help" for usage hints.
4Connected to a transient in-memory database.
5Use ".open FILENAME" to reopen on a persistent database.
6sqlite> create table users (id int, username varchar(255), email varchar(255));
7sqlite> .tables
8users
9sqlite> .exit
10~
我们的主函数将循环打印提示,获取输入行,然后处理该输入行:
1int main(int argc, char* argv[]) {
2 InputBuffer* input_buffer = new_input_buffer();
3 while (true) {
4 print_prompt();
5 read_input(input_buffer);
6
7 if (strcmp(input_buffer->buffer, ".exit") == 0) {
8 close_input_buffer(input_buffer);
9 exit(EXIT_SUCCESS);
10 } else {
11 printf("Unrecognized command '%s'.\n", input_buffer->buffer);
12 }
13 }
14}
我们将InputBuffer定义为一个小包装,用于包装需要与getline()[3]进行交互的状态。(稍后详细介绍)
1typedef struct {
2 char* buffer;
3 size_t buffer_length;
4 ssize_t input_length;
5} InputBuffer;
6
7InputBuffer* new_input_buffer() {
8 InputBuffer* input_buffer = (InputBuffer*)malloc(sizeof(InputBuffer));
9 input_buffer->buffer = NULL;
10 input_buffer->buffer_length = 0;
11 input_buffer->input_length = 0;
12
13 return input_buffer;
14}
接下来,print_prompt()函数会向用户打印提示。我们在读取每一行的输入之前执行此操作。
1void print_prompt() { printf("db > "); }
用getline()函数来读取输入:
1ssize_t getline(char **lineptr, size_t *n, FILE *stream);
lineptr:指向变量的指针,我们用它来指向读取行的缓存区。如果将其设置为NULL,它将由getline分配,由用户释放,即使命令失败。
N:指向用于保存分配的缓冲区大小的变量的指针。
流:要读取的输入流。我们按标准输入读取内容。
返回值:读取的字节数,可能小于缓冲区的大小。
我们用getline把read line的内容存储在inputbuffer->buffer同时把缓冲区的大小值存储在inputbuffer->bufferlength。我们把返回值存储在inputbuffer->input_length。
缓冲区初始为空,getline会保留足够的内存来存放输入的内容,并让缓冲区指向该存储空间。
1void read_input(InputBuffer* input_buffer) {
2 ssize_t bytes_read =
3 getline(&(input_buffer->buffer), &(input_buffer->buffer_length), stdin);
4
5 if (bytes_read <= 0) {
6 printf("Error reading input\n");
7 exit(EXIT_FAILURE);
8 }
9
10 // Ignore trailing newline
11 input_buffer->input_length = bytes_read - 1;
12 input_buffer->buffer[bytes_read - 1] = 0;
13}
这里我们定义一个函数,该函数释放为InputBuffer *实例分配的内存和相应结构的缓冲区元素(getline为readinput中的inputbuffer-> buffer分配内存)。
1void close_input_buffer(InputBuffer* input_buffer) {
2 free(input_buffer->buffer);
3 free(input_buffer);
4}
最后,我们解析并执行命令。现在只有一个可识别的命令:.exit,它用来终止程序。否则,我们将打印错误消息并继续循环。
1if (strcmp(input_buffer->buffer, ".exit") == 0) {
2 close_input_buffer(input_buffer);
3 exit(EXIT_SUCCESS);
4} else {
5 printf("Unrecognized command '%s'.\n", input_buffer->buffer);
6}
我们来执行一下:
1~ ./db
2db > .tables
3Unrecognized command '.tables'.
4db > .exit
5~
我们REPL一切正常。在下一部分中,我们将开始开发命令语言。同时,本章的所有代码如下:
1#include <stdbool.h>
2#include <stdio.h>
3#include <stdlib.h>
4#include <string.h>
5
6typedef struct {
7 char* buffer;
8 size_t buffer_length;
9 ssize_t input_length;
10} InputBuffer;
11
12InputBuffer* new_input_buffer() {
13 InputBuffer* input_buffer = malloc(sizeof(InputBuffer));
14 input_buffer->buffer = NULL;
15 input_buffer->buffer_length = 0;
16 input_buffer->input_length = 0;
17
18 return input_buffer;
19}
20
21void print_prompt() { printf("db > "); }
22
23void read_input(InputBuffer* input_buffer) {
24 ssize_t bytes_read =
25 getline(&(input_buffer->buffer), &(input_buffer->buffer_length), stdin);
26
27 if (bytes_read <= 0) {
28 printf("Error reading input\n");
29 exit(EXIT_FAILURE);
30 }
31
32 // Ignore trailing newline
33 input_buffer->input_length = bytes_read - 1;
34 input_buffer->buffer[bytes_read - 1] = 0;
35}
36
37void close_input_buffer(InputBuffer* input_buffer) {
38 free(input_buffer->buffer);
39 free(input_buffer);
40}
41
42int main(int argc, char* argv[]) {
43 InputBuffer* input_buffer = new_input_buffer();
44 while (true) {
45 print_prompt();
46 read_input(input_buffer);
47
48 if (strcmp(input_buffer->buffer, ".exit") == 0) {
49 close_input_buffer(input_buffer);
50 exit(EXIT_SUCCESS);
51 } else {
52 printf("Unrecognized command '%s'.\n", input_buffer->buffer);
53 }
54 }
55}
相关链接:





