InnoDB物理行中null值的存储的推断与验证
前言
想写这边文章,是因为之前想写一个解析innodb ibd文件的工具,在写这个工具的过程中,发现逻辑记录转物理记录的转换中,最难的有两部分,一是每行每字段null值占用的字节和存储,二是变长字段占用的字节和存储的格式。本文中重点针对第一种情况。第二种情况之后会专门写一篇
之前看有关介绍compact行记录格式:
变长字段之后的第二个部分是NULL标志位,该位指示了该行数据中是否有NULL值,有则用1表示。该部分所占字节为1字节
-----《InnoDB存储引擎》103页,作者:姜承尧
之后便思考是否不管有多少个列都是NULL,该部分都只占1个字节呢?
便有了如下测试
本文约定
逻辑记录:record (元组)
物理记录:row(行)
只讨论compact行格式
所用工具
自己python写的工具innodb_extract
测试数据
表结构
localhost.test>desc null_test;
+------------------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+------------------+--------------+------+-----+---------+----------------+
| id | bigint(20) | NO | PRI | NULL | auto_increment |
| name | varchar(20) | YES | | NULL | |
| legalname | varchar(25) | YES | | NULL | |
| industry | varchar(10) | YES | | NULL | |
| province | varchar(10) | YES | | NULL | |
| city | varchar(15) | YES | | NULL | |
| size | varchar(15) | YES | | NULL | |
| admin_department | varchar(128) | YES | | NULL | |
+------------------+--------------+------+-----+---------+----------------+
8 rows in set (0.00 sec)
表内数据
+----+------+-----------+----------+----------+------+------+------------------+
| id | name | legalname | industry | province | city | size | admin_department |
+----+------+-----------+----------+----------+------+------+------------------+
| 1 | NULL | NULL | NULL | NULL | NULL | NULL | NULL |
| 2 | TOM | NULL | NULL | NULL | NULL | NULL | NULL |
| 3 | ALEX | NULL | NULL | NULL | NULL | NULL | HR |
+----+------+-----------+----------+----------+------+------+------------------+
3 rows in set (0.00 sec)
分析数据
通过工具看三行数据
# python innodb_extract.py null_test.ibd
infimum
7f 000010001c 8000000000000001 0000f1e27b17 b5000001680084
1
7e 0000180020 8000000000000002 0000f1e27b17 b5000001680094 544f4d
2 TOM
3e 000020ffb6 8000000000000003 0000f1e27b17 b50000016800a4 414c4558 4852
3 ALEX HR
第一行:
null标志位:0x7f (01111111)
说明:从右向左方向写,一共7个null值
record header:000010001c
Transaction Id:0000f1e27b17
Roll Pointer:b5000001680084
数据:
第二行:
null标志位:0x7e (01111110)
说明:除第二列,其余均是null值
record header:0000180020
Transaction Id:0000f1e27b17
Roll Pointer:b5000001680084
数据:
第二列:544f4d => TOM
第三行:
null标志位:0x3e (00111110)
说明:除了第2列和第8列,其余均是null值
record header:000020ffb6
Transaction Id:0000f1e27b17
Roll Pointer:b5000001680084
数据:
第二列:414c4558 => ALEX
第八列:4852 => HR
假设
继续上面,如果包含Null值的字段是8个,或者9个会是怎样?
深度剖析
代码片段,该函数将物理记录转化为逻辑记录,版本5.5.31,源文件rem0rec.c,
rec_convert_dtuple_to_rec_comp(
/*===========================*/
rec_t* rec, /*!< in: origin of record */
const dict_index_t* index, /*!< in: record descriptor */
const dfield_t* fields, /*!< in: array of data fields */
ulint n_fields,/*!< in: number of data fields */
ulint status, /*!< in: status bits of the record */
ibool temp) /*!< in: whether to use the
format for temporary files in
index creation */
{
const dfield_t* field;
const dtype_t* type;
byte* end;
byte* nulls;
byte* lens;
ulint len;
ulint i;
ulint n_node_ptr_field;
ulint fixed_len;
ulint null_mask = 1;
ut_ad(temp || dict_table_is_comp(index->table));
ut_ad(n_fields > 0);
if (temp) {
ut_ad(status == REC_STATUS_ORDINARY);
ut_ad(n_fields <= dict_index_get_n_fields(index));
n_node_ptr_field = ULINT_UNDEFINED;
nulls = rec - 1;
if (dict_table_is_comp(index->table)) {
/* No need to do adjust fixed_len=0. We only
need to adjust it for ROW_FORMAT=REDUNDANT. */
temp = FALSE;
}
} else {
nulls = rec - (REC_N_NEW_EXTRA_BYTES + 1);
switch (UNIV_EXPECT(status, REC_STATUS_ORDINARY)) {
case REC_STATUS_ORDINARY:
ut_ad(n_fields <= dict_index_get_n_fields(index));
n_node_ptr_field = ULINT_UNDEFINED;
break;
case REC_STATUS_NODE_PTR:
ut_ad(n_fields
== dict_index_get_n_unique_in_tree(index) + 1);
n_node_ptr_field = n_fields - 1;
break;
case REC_STATUS_INFIMUM:
case REC_STATUS_SUPREMUM:
ut_ad(n_fields == 1);
n_node_ptr_field = ULINT_UNDEFINED;
break;
default:
ut_error;
return;
}
}
end = rec;
lens = nulls - UT_BITS_IN_BYTES(index->n_nullable);
/* clear the SQL-null flags */
memset(lens + 1, 0, nulls - lens);
结合COMPACT row格式来看:
row记录格式如下:
|---------------------extra_size-----------------------------------------|---------fields_data------------|
|--columns_lens---|---null lens----|------fixed_extrasize(5)-------------|--col1---|---col2---|---col2----|
|end<--------begin|end<-------beign|-------------------------------------|orgin---------------------------|
- 先看nulls = rec - (REC_N_NEW_EXTRA_BYTES + 1)
rec为记录开始的offset,也就是,extrasize也就是固定长度的record header的长度。注意null标志位和变长字段长度列表是从右->左的方向写的(原因可参见下部分代码)。所以nulls指向的是null lens
后一字节开始的位置。 - 再看lens = nulls - UT_BITS_IN_BYTES(index->n_nullable)
index->n_nullable指的是表结构中定义can be null的字段的个数,一个字段用一个bit来标记,UT_BITS_IN_BYTES将占用bit数转为占用的字节数。所以lens指向的是column_lens后面一个字节的位置,即跳过了Null标志的占用的空间,同样在写入值的时候也是从后面向前面写。 - memset(lens + 1, 0, nulls - lens) 将nulls空间清零。
之后就是遍历每一个字段,先对定义了can be null字段进行处理
/* Store the data and the offsets */
for (i = 0, field = fields; i < n_fields; i++, field++) {
const dict_field_t* ifield;
type = dfield_get_type(field);
len = dfield_get_len(field);
if (UNIV_UNLIKELY(i == n_node_ptr_field)) {
ut_ad(dtype_get_prtype(type) & DATA_NOT_NULL);
ut_ad(len == REC_NODE_PTR_SIZE);
memcpy(end, dfield_get_data(field), len);
end += REC_NODE_PTR_SIZE;
break;
}
if (!(dtype_get_prtype(type) & DATA_NOT_NULL)) {
/* nullable field */
ut_ad(index->n_nullable > 0);
if (UNIV_UNLIKELY(!(byte) null_mask)) {
nulls--;
null_mask = 1;
}
因为方向是从右向左写,也就是从后往前写,如果该字段为null,则将null标志位设为1并向前移1位,如果满了8个,也就是有8个字段都为null则offset向左移1位,并将null_mask置为1
==从这段代码看出之前的猜想,也就是并不是Null标志位只固定占用1个字节==,而是以8为单位,满8个null字段就多1个字节,不满8个也占用1个字节,高位用0补齐
ut_ad(*nulls < null_mask);
/* set the null flag if necessary */
if (dfield_is_null(field)) {
*nulls |= null_mask;
null_mask <<= 1;
continue;
}
null_mask <<= 1;
}
这段代码是就是设置null字段与null标志位的映射关系,如果字段为null,则设置标志位为1。
栗子验证
翻过来再看之前的例子,我们逐步的添加字段并设置default null看下null标志位的变化
- step 1,添加两个并设置default null
localhost.test>alter table null_test add column `kind` varchar(15) DEFAULT NULL after `size`;
Query OK, 3 rows affected (0.09 sec)
Records: 3 Duplicates: 0 Warnings: 0
localhost.test>alter table null_test add column licenseno varchar(15) DEFAULT NULL after `kind`;
Query OK, 3 rows affected (0.11 sec)
Records: 3 Duplicates: 0 Warnings: 0.11
那么理论来讲,第一行数据有9个null列了。满8个null列之后,继续向左写移,写1个bit之后开始占据两个字节。我们通过工具解析之后看下
# python innodb_extract.py null_test.ibd
01ff 000010001d 8000000000000001 0000f1e27c81 980000028c0084
1
01fe 0000180021 8000000000000002 0000f1e27c81 980000028c0094 544f4d
2 TOM
00fe 000020ffb3 8000000000000003 0000f1e27c81 980000028c00a4 414c455848
3 ALEX HR
第一行null标志位变为0x01ff,即00000001 11111111
一共有9个null字段,满了8位之后,继续向前占1个字节从右往左继续写
同理,第二行0x01fe,即00000001 11111110
第三行0x00fe,00000000 11111110
再继续添加8个字段并设置default null
localhost.test>desc null_test;
+------------------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+------------------+--------------+------+-----+---------+----------------+
| id | bigint(20) | NO | PRI | NULL | auto_increment |
| name | varchar(20) | YES | | NULL | |
| legalname | varchar(25) | YES | | NULL | |
| industry | varchar(10) | YES | | NULL | |
| province | varchar(10) | YES | | NULL | |
| city | varchar(15) | YES | | NULL | |
| size | varchar(15) | YES | | NULL | |
| kind | varchar(15) | YES | | NULL | |
| licenseno | varchar(15) | YES | | NULL | |
| admin_department | varchar(128) | YES | | NULL | |
| null_col1 | varchar(15) | YES | | NULL | |
| null_col2 | varchar(15) | YES | | NULL | |
| null_col3 | varchar(15) | YES | | NULL | |
| null_col4 | varchar(15) | YES | | NULL | |
| null_col5 | varchar(15) | YES | | NULL | |
| null_col6 | varchar(15) | YES | | NULL | |
| null_col7 | varchar(15) | YES | | NULL | |
| null_col8 | varchar(15) | YES | | NULL | |
+------------------+--------------+------+-----+---------+----------------+
18 rows in set (0.00 sec)
最多Null字段的第一行目前有个17个null字段,对应17个Null bit
root@hebe211 ibd]# python innodb_extract.py null_test.ibd
01ffff 000010001e 8000000000000001 0000f1e27cce c60000017600840301fffe0000
1
01fffe 0000180022 8000000000000002 0000f1e27cce c6000001760094 544f4d
2 TOM
01fefe 000020ffb0 8000000000000003 0000f1e27cce c60000017600a4 414c45 5848
3 ALEX HR
第一行null标志位变为0x01ff,即00000001 11111111 11111111
一共有17个null字段,满了两个8位之后,继续向前占1个字节从右往左继续写
同理,第二行0x01fe,即00000001 11111111 11111110
第三行0x00fe,00000001 11111110 11111110
结论
允许null的字段需要额外的空间来保存字段Null到null标志位映射的对应关系,所以保存这个映射关系的null标志位长度并不是固定的。也就是null字段越多并不是越省空间。实际生产环境中应尽量减少can be null的字段。