Mysql中的char与varchar length()与char_length()

在mysql中,char和varchar都表示字符串类型.但他们存储和检索数据的方式并不相同.

在表结构定义中声明char和varchar类型时,必须指定字符串的长度.也就是该列所能存储多少个字符(不是字节,是字符).例如:char(10)和varchar(10)都能存储10个字符.

声明为char的列长度是固定的,char的长度可选范围在0-255之间.也就是char最大能存储255个字符.如果该列是utf8编码,则该列所占用的字节数=字符数*3.如果是gbk编码则该列所占用的字节数=字符数*2.

声明为varchar的列长度是可变的,在mysql5.0.3之前varchar的长度范围为0-255,mysql5.0.3之后varchar的长度范围为0-65535个字节.采用varchar类型存储数据需要1-2个字节(长度超过255时需要2个字节)来存储字符串的实际长度.如果该列的编码为gbk,每个字符最多占用2个字节,最大长度不能超过32766个字符.如果该列的编码为utf8,每个字符最多占3个字节,最大字符长度为21845.

总结:char和varchar最大的不同就是一个是固定长度,一个是可变长度.由于是可变长度,因此存储的是实际字符串再加上一个记录字符串长度的字节。如果分配给char或varchar列的值超过列的最大长度,则对值进行裁剪.


length()与char_length()

length():Return the length of a string in bytes.返回字符串所占的字节数(受编码影响).

char_length():Return number of characters in argument.返回字符串中的字符数.


test:

1.建表

create table t_char_test(
id int(11) primary key,
name_char char(10) ,
name_varchar varchar(10) 
)engine=Myisam default charset=utf8;

insert into t_char_test(id,name_char,name_varchar) values (0,'字符类型','字符类型');

结果集如下:



<think>好的,我现在需要详细思考用户关于CHARVARCHAR区别的问题。用户之前已经询问过如何创建宿舍名单的SQL表结构,现在进一步探讨数据类型的选择,特别是CHARVARCHAR的区别。这说明用户可能在设计数据库时遇到了存储效率或字段长度的问题,或者想更深入理解这两种数据类型的适用场景。 首先,我需要回忆CHARVARCHAR的基本定义。CHAR是定长字符串,无论实际数据长度如何,都会占用定义的长度空间。例如,CHAR(5)存储"abc"会占用5个字符的空间,剩余部分用空格填充。而VARCHAR是变长字符串,根据实际数据长度分配存储空间,外加额外的字节来记录长度。比如VARCHAR(5)存储"abc"只会占用3个字符的空间加上长度信息。 接下来,要比较两者的存储方式。CHAR由于是定长,可能会有空间浪费,但读取速度快,因为长度固定,便于预计算存储位置。VARCHAR节省空间,但可能会有额外的开销,因为需要存储长度信息,并且在修改数据时可能导致行溢出,影响性能。 适用场景方面,CHAR适合存储长度相对固定且较短的字符串,比如性别、国家代码等。而VARCHAR适合长度变化较大的数据,如用户名、地址等。需要提醒用户注意数据库系统的具体实现,比如在某些系统中,VARCHAR可能有最大长度限制,或者在处理非常短的字符串时,CHAR可能更高效。 用户之前的问题中,他们使用了CHAR(5)来存储姓名,这可能存在问题,因为中文姓名可能超过5个字符,比如复姓或较长的名字。建议改用VARCHAR并适当调整长度,比如VARCHAR(10)或更长的,以适应不同的情况。这也是之前回复中提到的优化点,所以需要保持一致性。 此外,需要注意不同数据库系统对CHARVARCHAR的处理可能略有不同,比如是否自动去除末尾空格,最大长度限制等。例如,MySQLVARCHAR最多可以存储65535字节,但实际长度还受行大小限制。而CHAR在存储时会填充空格,检索时会自动去除,这可能导致一些意想不到的行为,比如比较时忽略末尾空格。 还需要考虑性能方面。如果表中的列多数为定长数据,使用CHAR可以让整个表成为静态表,提高查询效率。反之,如果有很多变长字段,使用VARCHAR节省空间,但可能影响查询速度。在内存和磁盘空间之间需要权衡。 可能用户没有考虑到字符集的影响,比如UTF-8下,每个字符可能占用多个字节,CHAR(5)可能会占用更多的存储空间。而VARCHAR在这种情况下,实际存储的字节数会根据字符而变化,更加灵活。 最后,需要以清晰的结构将这些点整理出来,确保用户能够逐步理解两者的区别,并应用到实际数据库设计中。同时,结合之前的例子,说明为何在姓名字段中使用VARCHAR更合适,加强用户的理解。</think>以下是关于 **CHAR** 和 **VARCHAR** 数据类型的详细对比分析: --- ### 一、核心区别总结 | **特性** | **CHAR** | **VARCHAR** | |----------------|-----------------------------------|---------------------------------| | **存储方式** | 定长字符串(固定分配空间) | 变长字符串(按需分配空间) | | **空格处理** | 自动填充空格至定义长度 | 不填充,保留原始长度 | | **存储效率** | 可能浪费空间(短数据时) | 更节省空间(适应不同长度) | | **查询速度** | 较快(长度固定,无需计算) | 稍慢(需读取长度信息) | | **适用场景** | 长度固定的短字符串(如编码、标志) | 长度变化较大的字符串(如描述、名称) | --- ### 二、存储机制详解 1. **CHAR** - 定义示例:`CHAR(5)` - 存储行为: - 存入"abc" ➔ 实际存储为`'abc '`(补2个空格) - 始终占用**5个字符**的存储空间 - 特点: - 读取时自动去除末尾空格 - 比较操作忽略末尾空格(`'a' = 'a '`返回`True`) 2. **VARCHAR** - 定义示例:`VARCHAR(5)` - 存储行为: - 存入"abc" ➔ 实际存储`'abc'`(占用3字符 + 长度信息1-2字节) - 最大占用空间 = 定义长度 + 额外长度标识(通常1-2字节) - 特点: - 保留所有空格(包括末尾) - 比较操作严格区分空格(`'a' ≠ 'a '`) --- ### 三、性能空间对比 - **空间效率**(假设字段定义为`CHAR(10)` vs `VARCHAR(10)`): | 输入值 | CHAR占用空间 | VARCHAR占用空间 | |-------------|-------------|-----------------| | "Hi" | 10字节 | 2字节 + 1字节长度标识 = **3字节** | | "Hello" | 10字节 | 5字节 + 1字节长度标识 = **6字节** | | "HelloWorld"| 10字节(截断)| 10字节 + 1字节长度标识 = **11字节**(若允许存储) | - **性能影响**: - `CHAR`在频繁查询的固定长度字段中更快(如状态码、性别) - `VARCHAR`在大文本或长度变化大的字段中更高效(如评论、地址) --- ### 四、使用场景建议 1. **优先使用 CHAR** - 字段值长度**完全固定**(如:`性别 ENUM('男','女')`) - 字段值**短且频繁查询**(如:`国家代码 CHAR(2)`) 2. **优先使用 VARCHAR** - 字段值长度**变化较大**(如:`姓名 VARCHAR(20)`) - 需要**节省存储空间**(如:长文本描述`VARCHAR(255)`) - 包含**末尾空格需保留**的场景(如:密码字段) --- ### 五、实际案例演示 ```sql -- 示例表 CREATE TABLE test ( fixed_code CHAR(3), -- 定长编码(如"CN"/"US") dynamic_desc VARCHAR(50) -- 变长描述 ); -- 插入数据 INSERT INTO test VALUES ('CN', '中国北京市'), ('US', '美国加利福尼亚州'); -- 查询结果 SELECT fixed_code, LENGTH(fixed_code) AS char_length, -- 始终返回3 LENGTH(dynamic_desc) AS varchar_length -- 返回实际长度 FROM test; ``` --- ### 六、注意事项 1. **字符集影响**: - UTF-8等变长编码下,`CHAR(n)`实际存储空间可能超过`n`字节(如中文占3字节/字符) 2. **数据库兼容性**: - 某些数据库(如Oracle)用`VARCHAR2`替代`VARCHAR`,需注意语法差异 3. **设计原则**: - 避免过度使用`CHAR`导致存储浪费 - `VARCHAR`最大长度需根据业务需求合理设定(如`VARCHAR(255)`是常见选择)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值