史上最简单的 MySQL 教程(十二)「记录长度」

温馨提示:本系列博文已经同步到 GitHub,地址为「mysql-tutorial」,欢迎感兴趣的童鞋StarFork,纠错。

记录长度

MySQL 中规定:任何一条记录最长不超过 65535 个字节,这意味着varchar永远达不到理论最大值

那么,varchar实际存储长度能达到多大呢?这由编码字符集决定。

下面,以varcharUTF-8GBK的情况为例,执行如下 SQL 语句,进行演示:

-- 求出 varchar 在 utf8 和 gbk 字符集下的实际最大值
create table my_utf8(
	name varchar(65535)
)charset utf8;

create table my_gbk(
	name varchar(65535)
)charset gbk;
1

观察上面的结果,发现咱们定义的字段name的长度超过限制啦,并且提示了其在utf8gbk字符集下各自的最大值。那么,咱们修改 SQL 语句如下,并再次执行:

-- 求出 varchar 在 utf8 和 gbk 字符集下的实际最大值
create table my_utf8(
	name varchar(21845)
)charset utf8;

create table my_gbk(
	name varchar(32767)
)charset gbk;
2

观察上面的执行结果,好吧,仍然在报错,为什么呢?观察如下 SQL 语句,并执行:

-- 求出 varchar 在 utf8 和 gbk 字符集下的实际最大值
create table my_utf8(
	name varchar(21844)  -- 21844 * 3 + 2 = 65534
)charset utf8;

create table my_gbk(
	name varchar(32766)  -- 32766 * 2 + 2 = 65534
)charset gbk;
3

如上图所示,咱们已经创建成功啦!至于什么定义字段name的长度为2184432766是由于:

  • 21845 * 3 + 2 = 65537 > 65535
  • 32767 * 2 + 2 = 65536 > 65535

因此,在提示的最大值的基础上各自减1. 至于,为什么还要加2,则是因为varchar为变长字符串,在其定义的时候,也就是说在分配存储空间的时候,都会自动多分配12个字节空间,因为咱们想要算最大的存储范围,所以加2.

在这里,细心的同学会发现一个问题,那就是:在咱们创建表my_utf8my_gbk的时候,咱们仅用了65534个字节,还剩余一个字节。现在,如果咱们想要将65535个字节都用了,怎么办呢?好说,增加一个tinyint类型的字段即可:

-- 求出 varchar 在 utf8 和 gbk 字符集下的实际最大值
create table my_utf81(
	stuno tinyint,       -- 1
	name varchar(21844)  -- 21844 * 3 + 2 = 65534
)charset utf8;

create table my_gbk1(
	stuno tinyint,       -- 1
	name varchar(32766)  -- 32766 * 2 + 2 = 65534
)charset gbk;
4

观察上面的结果,呃,竟然又出错啦!为什么啊?65534 + 1 = 65535,并没有超出范围啊!其实吧,之所以会出现这样的问题,是因为:在 MySQL 的记录中,如果有任何一个字段允许为空,那么系统就会自动从整个记录中保留一个字节来存储null,若想释放null所占的字节,则必须保证所有字段都不允许为空。

-- 求出 varchar 在 utf8 和 gbk 字符集下的实际最大值
create table my_utf82(
	stuno tinyint not null,       -- 1
	name varchar(21844) not null  -- 21844 * 3 + 2 = 65534
)charset utf8;

create table my_gbk2(
	stuno tinyint not null,       -- 1
	name varchar(32766) not null  -- 32766 * 2 + 2 = 65534
)charset gbk;
5

如上图所示,咱们已经成功创建了表my_utf82my_gbk2.

此外,在 MySQL 中,text文本字符串不占用记录长度,额外存储,但是text文本字符串也是属于记录的一部分,无论是在utf8还是在gbk字符集之中,其都占用记录中的10个字节长度,用来保存数据的地址以及长度。


———— ☆☆☆ —— 返回 -> 史上最简单的 MySQL 教程 <- 目录 —— ☆☆☆ ————

原创声明,本文系作者授权云+社区-专栏发表,未经许可,不得转载。

如有侵权,请联系 zhuanlan_guanli@qq.com 删除。

编辑于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏开源优测

python selenium2 - 鼠标键盘操作

完整路径 C:\Python27\Lib\site-packages\selenium\webdriver\common\action_chains.py 注...

3428
来自专栏枕边书

PHP用mb_string函数库处理与windows相关中文字符

昨天想批处理以前下载的一堆文件,把文件里的关键内容用正则匹配出来,集中处理。在操作文件时遇到一个问题,就是windows操作系统中的编码问题。 我们都知道win...

18510
来自专栏蓝天

sed 命令+正则表达式

sed是一个非交互性性文本编辑器, 它编辑文件或标准输入导出的文件拷贝。标准输入可能是来自键盘、文件重定向、字符串或变量,或者是一个管道文件。sed可以随意编辑...

752
来自专栏编程

正则表达式的游戏题目

游戏一(难度系数): 一个文本文件中有不少电话号码,它们的格式是用 1 开始的连续11位数字。比如:13923781654。现在为了规范,需要将他们转换为 13...

1837
来自专栏前端说吧

Sass-学习笔记【基础篇】

3354
来自专栏北京马哥教育

Python 基础语法

Python语言与Perl,C和Java等语言有许多相似之处。但是,也存在一些差异。 在本章中我们将来学习Python的基础语法,让你快速学会Python编程...

2996
来自专栏逆向技术

PE格式第七讲,重定位表

         PE格式第七讲,重定位表 一丶何为重定位(注意,不是重定位表格) 首先,我们先看一段代码,比如调用Printf函数,使用OD查看. ? 那么大...

1627
来自专栏散尽浮华

使用grep精确匹配一个单词

grep(global search regular expression(RE) and print out the line,全面搜索正则表达式并把行打印出...

925
来自专栏Django中文社区

Django模板标签regroup的妙用

在使用 Django 开发时,有时候我们需要在模板中按对象的某个属性分组显示一系列数据。例如博客文章按照时间归档分组显示文章列表(示例效果请看我的博客的归档页面...

2736
来自专栏cs

研究生的一份试题的几道题节选

首先祝朋友考研成功,勇往直前,我是不考研的,所以完全以提高能力,使用为主,不在意细节。小伙伴让我帮忙看了一下试卷,故截取了几道题目。 c我是真的应了那句话,从入...

3368

扫描关注云+社区