【MySQL】结构行长度的一些限制

今天被开发提交的DDL变更再次困惑,表中字段较多,希望将已有的两个varchar(4000)字段改为varchar(20000),我想innodb对varchar的存储不就是取前768字节记录当前行空间嘛,所以变更不会有任何问题的,但铁打的事实给了我结结实实的一个巴掌,直接报错,现在回放下这个错误!

模拟测试:

CREATE TABLE `ttt` (
    `id` DOUBLE ,
    `select_type` VARCHAR (57),
    `table` VARCHAR (192),
    `type` VARCHAR (30),
    `possible_keys` VARCHAR (22288),
    `key` VARCHAR (192),
    `key_len` VARCHAR (22288),
    `ref` VARCHAR (3072),
    `rows` DOUBLE ,
    `Extra` VARCHAR (765)
);
ERROR 1118 (42000): Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. This includes storage overhead, check the manual. You have to change some columns to TEXT or BLOBs

看了提示,表的2个varchar字段长度设置过长了,需要改成text,blob之类的类型,修改之后执行成功了。

mysql> use test;
Database changed
mysql>
mysql> CREATE TABLE `ttt` (
    -> `id` DOUBLE ,
    -> `select_type` VARCHAR (57),
    -> `table` VARCHAR (192),
    -> `type` VARCHAR (30),
    -> `possible_keys` TEXT,
    -> `key` VARCHAR (192),
    -> `key_len`  TEXT,
    -> `ref` VARCHAR (3072),
    -> `rows` DOUBLE ,
    -> `Extra` VARCHAR (765)
    -> );
Query OK, 0 rows affected (0.00 sec)

疑惑:varchar(N),这个N不是最大为65535吗?为什么设置成12288就会报错?12288比65535小很多啊。

上官网:http://dev.mysql.com/doc/refman/5.6/en/column-count-limit.html:

Every table (regardless of storage engine) has a maximum row size of 65,535 bytes. Storage engines may place additional constraints on this limit, reducing the effective maximum row size.

解析:65,535所说明的是针对的是整个表的非大字段类型的字段的bytes总合。

欲看详细分析还要继续往下看

每个表有4096个列的硬性限制,但是到具体表是往往小于这个数字,确切的限制取决于几个相互作用的因素:

  1. 每个表(不考虑存储引擎)为65,535字节的最大行大小限制。存储引擎可能会对这个限制进行额外的限制,降低了有效的最大行大小。

    • 受到行大小限制,列的数目还要看具体的字段长度,例如,UTF8字符需要三个字节存储,因此对于CHAR(255)CHARACTER SET UTF8列,服务器必须分配255×3 =765的字节。因此,一个表不能包含超过65,535/765=85这样的列。
    • 可变长度列在评估字段大小时还要考虑存储列实际长度的字节数。例如,VARCHAR(255)CHARACTER SET UTF8列需要额外的两个字节来存储值长度信息,所以该列需要多达767个字节存储,其实最大可以存储65533字节,剩余两个字节存储长度信息。
    • BLOB和TEXT列不同于varchar字段,列长度信息独立于行长存储,可以达到65535字节真实存储。
    • 声明NULL列可降低允许的最大列数。对于MyISAM表,NULL列需要该行中额外的空间记录其值是否为NULL。每个NULL列需要一个额外的位,四舍五入到最接近的字节。
    最大行长度计算如下:
        row length = 1
                + (sum of column lengths)
                + (number of NULL columns + delete_flag + 7)/8
                + (number of variable-length columns)

    对于静态表,delete_flag = 1,静态表通过在该行记录一个位来标识该行是否已被删除。 动态表时delete_flag = 0,因为该标记存储在动态行首,动态表具体可以根据row_format判断,详情参考Section 15.2.3, “MyISAM Table Storage Formats”

    对于InnoDB表,NULL和NOT NULL列存储大小是一样,因此上述计算并不适用。

    以下测试是没有问题的,(32765 + 2 + 32766 + 2 )bytes < 655535

    mysql> CREATE TABLE t1
        -> (c1 VARCHAR(32765) NOT NULL, c2 VARCHAR(32766) NOT NULL)
        -> ENGINE = MyISAM CHARACTER SET latin1;
    Query OK, 0 rows affected (0.02 sec)

    因为NULL属性需要额外的储存空间,超过了最大65535的限制

    mysql> CREATE TABLE t2
         -> (c1 VARCHAR(32765) NULL, c2 VARCHAR(32766) NULL)
         -> ENGINE = MyISAM CHARACTER SET latin1;
     ERROR 1118 (42000): Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. You have to change some
    columns to TEXT or BLOBs

    因为变成字段需要占用行内额外的储存空间,所以超过了最大65535的限制

    mysql> CREATE TABLE t3
        -> (c1 VARCHAR(65535) NOT NULL)
        -> ENGINE = MyISAM CHARACTER SET latin1;
    ERROR 1118 (42000): Row size too large. The maximum row size for the
    used table type, not counting BLOBs, is 65535. You have to change some
    columns to TEXT or BLOBs
  2. 除此之外,一些存储引擎可能会强加限制表中列数的附加限制。如:
    • InnoDB允许单表最多1000个列
    • InnoDB限制行大小不到数据库页面的一半,不包括VARBINARY,VARCHAR,BLOB或TEXT列。欲了解更多信息,更多影响体现DML,而不是DML。详情参考Limits on InnoDB Tables
    • InnoDB不同的存储格式(压缩,冗余)使用不同数量的页面头部和尾部,这会影响可用于存储行的长度。
    • 如果innodbstrictmode禁用,创建表时使用冗余(REDUNDANT)或紧凑COMPACT格式,如果列超过最大行大小也会成功的定义,只是产生警告:
      | Warning |  139 | Row size too large (> 8123). Changing some columns to TEXT or BLOB
      or using ROW_FORMAT=DYNAMIC or ROW_FORMAT=COMPRESSED may help.
      In current row format, BLOB prefix of 768 bytes is stored inline.
    • 如果innodbstrictmode禁用,创建表时使用动态(DYNAMIC)或压缩(COMPRESSED)格式,如果列超过最大行大小也会成功的定义,就会直接报错:
      ERROR 1118 (42000): Row size too large (> 8126). Changing some columns to TEXT or BLOB may help.
      In current row format, BLOB prefix of 0 bytes is stored inline.
  3. 每个表都有一个包含表定义的.frm文件。文件定义的内容也会影响到字段数的上限,详情参考Limits Imposed by .frm File Structure

参考:

时间: 2024-10-18 05:36:35

【MySQL】结构行长度的一些限制的相关文章

【MySQL】行长度的一些限制

今天开发在导入数据的时候报一个错误: Row size too large. The maximum row size for the used table type, not counting BLOBs, is 8126. You have to change some columns to TEXT or BLOBs 原因:发现导入的数据单行长度较长. 官方文档的解决办法为: 1.垂直表字段拆分或者大字段合并(大字段最多不超过768,业务进行合并+拆分),divide your table

MySQL 如何存储长度较大的varchar与blob

本文同时发表在https://github.com/zhangyachen/zhangyachen.github.io/issues/96 最近,在工作中遇到了MySQL中如何存储长度较长的字段类型问题,于是花了一周多的时间抽空学习了一下,并且记录下来. MySQL大致的逻辑存储结构在这篇文章中有介绍,做为基本概念:InnoDB 逻辑存储结构 注:文中所指的大数据指的是长度较长的数据字段,包括varchar/varbinay/text/blob. Compact行格式 我们首先来看一下行格式为C

[转]分析MySQL数据类型的长度【mysql数据字段 中length和decimals的作用!熟悉mysql必看】

你看到的这个文章来自于http://www.cnblogs.com/ayanmw 转载自:http://blog.csdn.net/daydreamingboy/article/details/6310907 分析MySQL数据类型的长度 MySQL有几种数据类型可以限制类型的"长度",有CHAR(Length).VARCHAR(Length).TINYINT(Length).SMALLINT(Length).MEDIUMINT(Length).INT(Length).BIGINT(L

mysql命令行参数

一,mysql命令行参数 Usage: mysql [OPTIONS] [database] //命令方式 -?, --help //显示帮助信息并退出 -I, --help //显示帮助信息并退出 --auto-rehash //自动补全功能,就像linux里面,按Tab键出提示差不多,下面有例子 -A, --no-auto-rehash //默认状态是没有自动补全功能的.-A就是不要自动补全功能 -B, --batch //ysql不使用历史文件,禁用交互 (Enables --silent

Mysql命令行基本操作

本文记录的是windows7系统下Mysql的命令行基本操作. 一.Mysql启动与登录 1. Mysql启动: 通过运行mysql安装目录下的/bin/mysqld.exe文件来启动Mysql服务.在命令行中进入mysql安装目录下的bin目录,执行命令: mysqld --defaults-file="C:\Program Files\MySQL\MySQL Server X.Y\my.ini"(指定配置文件,配置文件的位置根据具体情况而定) 即可启动mysql服务.为了以后开启m

mysql 命令行参数

MySQL命令行参数 Usage: mysql [OPTIONS] [database] //命令方式  例如: mysql -h${HOSTNAME}  -P${PORT}  -u${USERNAME} -p${PASSWORD}  -D ${DBNAME} -e  "${create_table_sql}" -?, --help //显示帮助信息并退出 -I, --help //显示帮助信息并退出 --auto-rehash //自动补全功能,就像linux里面,按Tab键出提示差

通过mysql命令行理解mysql

引言:工具不可谓给我们的生活带来了便利,但有些时候我们却忘记了事物本身的意义.在大多数人都在追捧甚至是盲从各种各样的工具有多先进的时候,你是否有反思过:你目前是否有使用它的资格. 假设你学会了使用一款软件,那么你仅仅是会一款软件而已,但是只有透过现象看本质,你才能理解它真正的意义. 追溯 现在让时间追溯到mysql刚开发出来的时候,那个时候没有phpmyadmin,没有navicat等等这样的mysql可视化管理工具,那么我们要如何来操作mysql? 使用mysql命令行或许比可视化工具来的要复

MySQL的varchar长度问题

From: http://blog.csdn.net/longyulu/article/details/7863737 http://dinglin.iteye.com/blog/914276 http://www.cnblogs.com/fakis/archive/2011/03/07/1976532.html 如果某一项中设置的是varchar(50) 那么对英文当然是50 那么对中文呢 utf-8的中文占3个字节 那么,这个varchar(50)是不是只能存16个汉字了? 不是这样的,my

MySql命令行命令和SQL语句

一.常用mysql命令行命令 1.启动MYSQL服务 net start mysql 停止MYSQL服务 net stop mysql 2.netstat -na|findstr 3306 查看被监听的端口,findstr用于查找后面的端口是否存在 3.在命令行中登录MYSQL控制台,即使用MYSQL COMMEND LINE TOOL 语法格式 mysql -user=root  -password=123456  db_name 或者简写格式 mysql -uroot -p123456 db