服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Mysql - MySQL 表空间碎片的概念及相关问题解决

MySQL 表空间碎片的概念及相关问题解决

2021-06-26 20:33王文安 Mysql

这篇文章主要介绍了MySQL 表空间碎片的概念及相关问题解决,帮助大家更好的理解和学习使用MySQL,感兴趣的朋友可以了解下

背景

经常使用 mysql 的话,会发现 mysql 数据文件的磁盘空间一般会不停的增长,而且有时候删了数据或者插入一批数据的时候,磁盘空间有时候还会毫无变化。引发这个其妙现象的就是 mysql 的表空间碎片。

什么是表空间碎片?

表空间碎片指的是表空间中存在碎片,形象一点来比喻的话,就像是一张 a4 纸,“表空间碎片”就像是把这张 a4 纸撕碎,再重新拼起来,各个碎片之间都会有一些缝隙存在,这些缝隙就是“表空间碎片”。重新拼起来的碎片实际上会比完整的 a4 纸大上一圈,这也代表着表空间容易引发的问题:空间浪费。

对于背景中描述的现象,可以用一张图来进行解释:

MySQL 表空间碎片的概念及相关问题解决

图中的数字代表真实的数据行,圆角矩形代表一个表的表空间。从左往右,第一次操作是删除数据,由于 mysql 在设计上是不会主动释放空间的,因此当表中的数据行被删除时,虽然数据被“删除”了,但是实际上这部分空间是没有释放的,依旧会被 table a 占用,因此也就出现了这样子的情景:删除了日志表的很多数据,但是 mysql 的磁盘空间并没有降低。

ps:这种不释放空间的设计多半和惰性删除有关,早期设计数据库时,使用的 io 设备一般是机械盘,读写性能比 ssd 差很多,所以删除操作一般不会直接触发磁盘上的数据删除。

可以看到数据删除之后,原本连续的空间中出现了两个空白的区域,这种一般就叫做表空间空洞,空洞太多了就叫做表空间碎片化(对应的是表空间连续)。这部分的空间虽然不会释放,但是会被标记为可重复利用,参考最右边的表空间示意图(第三个圆角矩形),当新插入数据的时候新数据会重新写入到表空间空洞中,这也代表着:在大规模删除过数据的表上,写入数据时,表空间可能不会明显增长或者不会增长。

实际上产生表空间空洞的操作并不只有 delete,update 也会引起这个问题,比如在 varchar 这种变长的字符型列中修改数据,改短一些的时候就会出现非常小的空洞,改长的话就有可能会因为空间不足导致把数据行的一些数据迁移到其他地方去。

怎么查看表空间碎片

mysql 的系统表记录了表空间的使用情况,可以用如下查询检查:

?
1
2
3
4
5
6
7
8
9
10
select concat(table_schema,'.',table_name) as 'table_name',
                table_rows as 'number of rows',
                concat(round(data_length/(1024*1024),2),' m') as 'data_size',
                concat(round(index_length/(1024*1024),2),' m') as 'index_size' ,
                concat(round(data_free/(1024*1024),2),' m') as'data_free',
                concat(round(data_free/data_length,2),' %') as 'data_free_pct',
                engine as 'engine'
from information_schema.tables
where table_schema = 'tablename'
order by data_free desc;

data_free 指表空间碎片的总空间大小,data_free_pct 指这个表的碎片百分比,效果如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
mysql> select concat(table_schema,'.',table_name) as 'table_name',
    ->                 table_rows as 'number of rows',
    ->                 concat(round(data_length/(1024*1024),2),' m') as 'data_size',
    ->                 concat(round(index_length/(1024*1024),2),' m') as 'index_size' ,
    ->                 concat(round(data_free/(1024*1024),2),' m') as'data_free',
    ->                 concat(round(data_free/data_length,2),' %') as 'data_free_pct',
    ->                 engine as 'engine'
    -> from information_schema.tables
    -> where table_schema = 'sbtest'
    -> order by data_free desc;
+----------------+----------------+-----------+------------+-----------+---------------+--------+
| table_name     | number of rows | data_size | index_size | data_free | data_free_pct | engine |
+----------------+----------------+-----------+------------+-----------+---------------+--------+
| sbtest.sbtest5 |              0 | 0.02 m    | 0.00 m     | 44.00 m   | 2816.00 %     | innodb |
| sbtest.sbtest4 |         986400 | 214.70 m  | 15.52 m    | 4.00 m    | 0.02 %        | innodb |
| sbtest.sbtest3 |         986400 | 214.70 m  | 15.52 m    | 4.00 m    | 0.02 %        | innodb |
| sbtest.sbtest2 |         986400 | 214.70 m  | 15.52 m    | 4.00 m    | 0.02 %        | innodb |
| sbtest.sbtest1 |         987400 | 199.70 m  | 15.52 m    | 4.00 m    | 0.02 %        | innodb |
+----------------+----------------+-----------+------------+-----------+---------------+--------+
5 rows in set (0.00 sec)

第一行数据是测试用的数据,表中的所有数据都被删掉了,因此计算出来的 data_free_pct 超过了 100%。

怎么解决表空间碎片问题

目前,能够回收表空间的办法仅有一个,就是重建表,手段包括但不限于 optimize,alter table 等。alter table 的有些操作只能靠 rebuild 表来完成,所以有时候对大表进行一些维护操作之后,也会看到磁盘空间使用率下降,这就是回收了表空间碎片腾出来的那一部分空间。

从一般经验来看,表空间碎片的回收操作不建议经常执行,每个月一次就足够了,因为 rebuild 表对服务器的资源影响会比较大,且会影响这个表的写入操作。碎片率(data_free_pct)低于 20% 的时候也不用特别在意,除非磁盘空间非常紧张,且日志基本被清空。

对于回收空间的问题

对一些日志表,或者是有区域性特征的表,建议使用 mysql 的分区表来管理,需要清理一批数据的时候,可以用 partition truncate 的方式进行清理,磁盘空间也能直接释放掉。

以上就是mysql 表空间碎片的概念及相关问题解决的详细内容,更多关于mysql 表空间碎片的资料请关注服务器之家其它相关文章!

延伸 · 阅读

精彩推荐
  • Mysql解决MySQl查询不区分大小写的方法讲解

    解决MySQl查询不区分大小写的方法讲解

    今天小编就为大家分享一篇关于解决MySQl查询不区分大小写的方法讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起...

    Veir_dev5592019-06-25
  • MysqlMySQL数据库varchar的限制规则说明

    MySQL数据库varchar的限制规则说明

    本文我们主要介绍了MySQL数据库中varchar的限制规则,并以一个实际的例子对限制规则进行了说明,希望能够对您有所帮助。 ...

    mysql技术网4192019-11-23
  • MysqlMySQL锁的知识点总结

    MySQL锁的知识点总结

    在本篇文章里小编给大家整理了关于MySQL锁的知识点总结以及实例内容,需要的朋友们学习下。...

    别人放弃我坚持吖4362020-12-14
  • Mysql浅谈mysql 树形结构表设计与优化

    浅谈mysql 树形结构表设计与优化

    在诸多的管理类,办公类等系统中,树形结构展示随处可见,本文主要介绍了mysql 树形结构表设计与优化,具有一定的参考价值,感兴趣的小伙伴们可以参...

    小码农叔叔5242021-11-16
  • Mysqlmysql 不能插入中文问题

    mysql 不能插入中文问题

    当向mysql5.5插入中文时,会出现类似错误 ERROR 1366 (HY000): Incorrect string value: '\xD6\xD0\xCE\xC4' for column ...

    MYSQL教程网5722019-11-25
  • MysqlERROR: Error in Log_event::read_log_event()

    ERROR: Error in Log_event::read_log_event()

    ERROR: Error in Log_event::read_log_event(): read error, data_len: 438, event_type: 2 ...

    MYSQL教程网6402020-03-13
  • Mysql详解MySQL中的分组查询与连接查询语句

    详解MySQL中的分组查询与连接查询语句

    这篇文章主要介绍了MySQL中的分组查询与连接查询语句,同时还介绍了一些统计函数的用法,需要的朋友可以参考下 ...

    GALAXY_ZMY5432020-06-03
  • MysqlMySQL 数据备份与还原的示例代码

    MySQL 数据备份与还原的示例代码

    这篇文章主要介绍了MySQL 数据备份与还原的相关知识,本文通过示例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下...

    逆心2962019-06-23