MySQL表空间分system/file-per-table/general三类:system即ibdata1,存系统数据;file-per-table每表独立.ibd;general需手动创建共享表空间。

MySQL 的表空间不是抽象概念,而是对应磁盘上真实的数据文件。关键区别在 innodb_file_per_table 配置和 CREATE TABLE 时是否显式指定 TABLESPACE。
默认开启 innodb_file_per_table=ON(MySQL 5.6.6+),此时每张 InnoDB 表生成独立的 .ibd 文件;若关闭,则所有表数据都写入共享表空间 ibdata1 —— 这会导致 DROP TABLE 后空间无法回收,且无法用 OPTIMIZE TABLE 收缩单表。
SYSTEM 表空间:固定为 ibdata1(及可能的 ibdata2 等),存储数据字典、undo log、系统表等,不可删除FILE_PER_TABLE 表空间:每个表一个 .ibd,路径在 datadir/数据库名/表名.ibd,支持 TRUNCATE 和 OPTIMIZE 回收空间GENERAL 表空间:由 CREATE TABLESPACE ... ADD DATAFILE 创建,可跨库共享,但需手动管理路径与权限,且不支持临时表直接查 INFORMATION_SCHEMA.INNODB_SYS_TABLES 最可靠,SPACE 字段值为 0 表示 system 表空间,非 0 则是 file-per-table 或 general 表空间。
SELECT NAME, SPACE, FILE_FORMAT, ROW_FORMAT FROM INFORMATION_SCHEMA.INNODB_SYS_TABLES WHERE NAME = 'testdb/users';
再结合 INFORMATION_SCHEMA.FILES 查物理路径:
SELECT TABLESPACE_NAME, FILE_NAME, TOTAL_EXTENTS FROM INFORMATION_SCHEMA.FILES WHERE TABLESPACE_NAME = 'testdb/users' OR TABLESPACE_NAME = 'innodb_system';
注意:FILES 视图只显示已打开的数据文件,冷备后未重启 MySQL 可能看不到新 .ibd。
不是所有场景都适合用 GENERAL 表空间。它主要解决多表共用一个大文件、统一管理 I/O 调度的需求,但会引入额外约束:
ADD DATAFILE 指定的路径对 mysqld 进程可读写ALTER TABLE t1 TABLESPACE = my_ts 会重建整张表,期间锁表(online DDL 在 8.0.12+ 支持部分场景无锁,但表空间迁移仍需 copy algorithm).ibd 文件被移走,原路径只剩一个空壳,误删会导致表不可访问不能在线 shrink。因为 ibdata1 是 SYSTEM 表空间,包含 undo logs、数据字典等核心结构,MySQL 不提供收缩接口。常见错误操作是直接清空或删除 ibdata1,这会导致实例启动失败。
真正可行的路径只有两个:
innodb_undo_tablespaces 且 undo 存在独立表空间中,可通过 SET GLOBAL innodb_undo_log_truncate=ON 配合 innodb_max_undo_log_size 自动截断旧 undomysqldump --all-databases),停库,删除 ibdata1 + ib_logfile*,重启初始化新实例,再导入 —— 这是唯一能让 ibdata1 回到初始大小的方式所以一开始就把 innodb_file_per_table=ON 设为强制策略,比后期补救重要得多。