牛刀小试MySQL学习-String Types

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS PostgreSQL,高可用系列 2核4GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: Storage Requirements for String TypesData TypeStorage RequiredCHAR(M)M × w bytes, 0  M  255, where w is the number of bytes required ...
Storage Requirements for String Types
Data Type Storage Required
CHAR(M) M × w bytes, 0  M  255, where w is the number of bytes required for the maximum-length character in the character set
BINARY(M) M bytes, 0  M  255
VARCHAR(M), VARBINARY(M) L + 1 bytes if column values require 0 – 255 bytes, L + 2 bytes if values may require more than 255 bytes
TINYBLOB, TINYTEXT L + 1 bytes, where L 8
BLOB, TEXT L + 2 bytes, where L 16
MEDIUMBLOB, MEDIUMTEXT L + 3 bytes, where L 24
LONGBLOB, LONGTEXT L + 4 bytes, where L 32
ENUM('value1','value2',...) 1 or 2 bytes, depending on the number of enumeration values (65,535 values maximum)
SET('value1','value2',...) 1, 2, 3, 4, or 8 bytes, depending on the number of set members (64 members maximum)

char&varchar:

一个是定长字符类型,一个是可变长字符类型。char(M)(0~255),存储需求:M*w bytes.意思就是说:这个w嘛,它还得根据字符集,例如latin字符集

是1字节,gb2312是2字节,utf8是三字节。所以,create table t3(name char(255)) character set utf8;这name列存储的应该是255*3的字节。


varchar则是可变存储数据(0~65535),可以节省存储空间。但它有个前缀长度1字节,这个基本上可以猜测它用来定义可以长度的标识量。因为1字节=>8bit=>256

所以,varchar超过255字节后,前缀长度就得使用两个字节了。

以下表格,是官方给出的char和varchar对应的存储字节。

Value CHAR(4) Storage Required VARCHAR(4) Storage Required
'' '    ' 4 bytes '' 1 byte
'ab' 'ab  ' 4 bytes 'ab' 3 bytes
'abcd' 'abcd' 4 bytes 'abcd' 5 bytes
'abcdefgh' 'abcd' 4 bytes 'abcd' 5 bytes

char(4)就一直是固定长度

varchar(4)就是可变长度,前面有1字节的前缀长度,超过255字节,就是2字节的前缀长度


性能的比较:由于,char是固定长度的,所以是用空间来换取时间,所以char的查询速度是比varchar的查询速度要快的。

但是对于Innodb存储引擎来说:内部的行存储就本身没有区分固定长度和可变长度列(数据行是使用了头指针,来指向数据列值的),所以这里

使用varchar更好,性能也不比char差


BINARY&VARBINARY:

BINARY和VARBINARY类似于CHAR和VARCHAR,不同的是她们包含二进制字符串而不包含非二进制字符串。

mysql> create table t(C binary(3));

Query OK, 0 rows affected (0.15 sec)


mysql> insert into t values('a');

Query OK, 1 row affected (0.05 sec)


mysql> select * from t;

+------+

| C    |

+------+

| a    |

+------+

1 row in set (0.00 sec)


mysql> select *,hex(c),c='a',c='a\0',c='a\0\0' from t;

+------+--------+-------+---------+-----------+

| C    | hex(c) | c='a' | c='a\0' | c='a\0\0' |

+------+--------+-------+---------+-----------+

| a    | 610000 |     0 |       0 |         1 |

+------+--------+-------+---------+-----------+

1 row in set (0.02 sec)

从上面可以看出,它在保存a值的时候,会在最后默认填充'0x00'(零字节),所以c='a\0\0' boolean返回的值就是1了。


TEXT&BLOB

TEXT保存较大的文本,BLOB保存二进制数据。存储的形式上面的表格有描述。

BLOB和TEXT的数据在被删除时,会引起"空洞"的记录。但是innodb可以阻止这个情况的发生

mysql> create table t8(id int,context text);

mysql> insert into t10 values(1,repeat('zsdzsd',100));(重复插入到,有几十万条数据)

mysql> insert into t10 select * from t10;

查看ibd

[root@localhost test]# du -sh t10.*

12K     t10.frm

269M    t10.ibd

和ibdata1

[root@localhost innodb_ts]# du -sh ./*

2.1G    ./ibdata1

再把数据删除。

mysql> delete from t10 where id = 1;

Query OK, 131072 rows affected (37.98 sec)

再次查看ibd和ibdata1,发现大小没变,这里就出现了空洞


在innodb存储引擎中,是无法使用optimize的,需要使用recreate和analyze代替。

mysql> optimize table t10;

+----------+----------+----------+-------------------------------------------------------------------+

| Table    | Op       | Msg_type | Msg_text                                                          |

+----------+----------+----------+-------------------------------------------------------------------+

| test.t10 | optimize | note     | Table does not support optimize, doing recreate + analyze instead |

| test.t10 | optimize | status   | OK                                                                |

+----------+----------+----------+-------------------------------------------------------------------+

2 rows in set (2 min 36.33 sec)

虽然这样看起来,像是造成了空洞,但是如果你在插入一张表

mysql> create table t11(name varchar(20));

Query OK, 0 rows affected (0.08 sec)


mysql> insert into t11 values(repeat('zsd',10));

Query OK, 1 row affected, 1 warning (0.04 sec)


mysql> insert into t11 select * from t11;(继续重复插入50w条数据)

Query OK, 1 row affected (0.04 sec)

Records: 1  Duplicates: 0  Warnings: 0

再去查看系统层面的大小:(发现,数据已经被释放了,t11正好用上了t10释放的空间,但是ibddata还是那么大)

[root@localhost innodb_ts]# du -sh ./*

2.1G    ./ibdata1

[root@localhost test]# du -sh t11.*

12K     t11.frm

61M     t11.ibd

[root@localhost test]# du -sh t10.*

12K     t10.frm

181M    t10.ibd

关于排序的问题:TEXT和BLOB排序的时候,会用上临时表,这样对性能来说是有下降的。

使用合成的(Synthetic)索引,可以提高大文本字段的查询性能,适合精确查询,不适合模糊查询。

create table t12(id int,context text,hash_value varchar(40));

insert into t12 values(1,  repeat ('zsd',2),md5(context));

insert into t12 values(2,  repeat ('zsd',3),md5(context));

insert into t12 values(3,  repeat ('zsd2008',3),md5(context));

mysql> select * from t12;

+------+-----------------------+----------------------------------+

| id   | context               | hash_value                       |

+------+-----------------------+----------------------------------+

|    1 | zsdzsd                | 0a8a135e9ae7d104ae7a81f4733b66cd |

|    2 | zsdzsdzsd             | 912d5bffb559146700698871e7730447 |

|    3 | zsd2008zsd2008zsd2008 | 93d60b10af275780ff14ce2de0e20445 |

+------+-----------------------+----------------------------------+

3 rows in set (0.07 sec)

mysql> select * from t12 where hash_value=md5('zsdzsd');

+------+---------+----------------------------------+

| id   | context | hash_value                       |

+------+---------+----------------------------------+

|    1 | zsdzsd  | 0a8a135e9ae7d104ae7a81f4733b66cd |

+------+---------+----------------------------------+

1 row in set (0.00 sec)


ENUM&SET

ENUM:对1~255各成员,存储一个字节,对于255~65535个成员,存储两个字节。

SET:1~8个成员:占1字节

          9~16成员:占2字节

          17~24成员:占3个字节

          25~32成员:占4个字节

          33~64成员:占8个字节

   

例子:create table t13(gender enum('M','F'));

           insert into t13 values('M'),('1'),('f'),(NULL);其他的字会默认变为M,NULL值还是NULL,忽略大小写

      

           create table t14(col set('a','b','c','d'));

           insert into t14 values('a,b'),('a,b,a'),('a,b'),('a,c'),('a'),('a,b,c,d');(就是可以组合自由搭配的存储到表中)

相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。   相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情: https://www.aliyun.com/product/rds/mysql 
相关文章
|
2月前
|
NoSQL 算法 Redis
【Docker】(3)学习Docker中 镜像与容器数据卷、映射关系!手把手带你安装 MySql主从同步 和 Redis三主三从集群!并且进行主从切换与扩容操作,还有分析 哈希分区 等知识点!
Union文件系统(UnionFS)是一种**分层、轻量级并且高性能的文件系统**,它支持对文件系统的修改作为一次提交来一层层的叠加,同时可以将不同目录挂载到同一个虚拟文件系统下(unite several directories into a single virtual filesystem) Union 文件系统是 Docker 镜像的基础。 镜像可以通过分层来进行继承,基于基础镜像(没有父镜像),可以制作各种具体的应用镜像。
425 5
|
3月前
|
关系型数据库 MySQL 数据管理
Mysql基础学习day03-作业
本内容包含数据库建表语句及多表查询示例,涵盖内连接、外连接、子查询及聚合统计,适用于员工与部门数据管理场景。
82 1
|
3月前
|
SQL 关系型数据库 MySQL
Mysql基础学习day01
本课程为MySQL基础学习第一天内容,涵盖MySQL概述、安装、SQL简介及其分类(DDL、DML、DQL、DCL)、数据库操作(查询、创建、使用、删除)及表操作(创建、约束、数据类型)。适合初学者入门学习数据库基本概念和操作方法。
189 6
|
3月前
|
SQL 关系型数据库 MySQL
Mysql基础学习day02-作业
本教程介绍了数据库表的创建与管理操作,包括创建员工表、插入测试数据、删除记录、更新数据以及多种查询操作,涵盖了SQL语句的基本使用方法,适合初学者学习数据库操作基础。
97 0
|
3月前
|
SQL 关系型数据库 MySQL
Mysql基础学习day03
本课程为MySQL基础学习第三天内容,主要讲解多表关系与多表查询。内容涵盖物理外键与逻辑外键的区别、一对多、一对一及多对多关系的实现方式,以及内连接、外连接、子查询等多表查询方法,并通过具体案例演示SQL语句的编写与应用。
106 0
|
3月前
|
SQL 关系型数据库 MySQL
Mysql基础学习day01-作业
本教程包含三个数据库表的创建练习:学生表(student)要求具备主键、自增长、非空、默认值及唯一约束;课程表(course)定义主键、非空唯一字段及数值精度限制;员工表(employee)包含自增主键、非空字段、默认值、唯一电话号及日期时间类型字段。每个表的结构设计均附有详细SQL代码示例。
86 0
|
3月前
|
SQL 关系型数据库 MySQL
Mysql基础学习day02
本课程为MySQL基础学习第二天内容,涵盖数据定义语言(DDL)的表查询、修改与删除操作,以及数据操作语言(DML)的增删改查功能。通过具体SQL语句与实例演示,帮助学习者掌握MySQL表结构操作及数据管理技巧。
152 0
|
12月前
|
SQL 存储 关系型数据库
【MySQL基础篇】全面学习总结SQL语法、DataGrip安装教程
本文详细介绍了MySQL中的SQL语法,包括数据定义(DDL)、数据操作(DML)、数据查询(DQL)和数据控制(DCL)四个主要部分。内容涵盖了创建、修改和删除数据库、表以及表字段的操作,以及通过图形化工具DataGrip进行数据库管理和查询。此外,还讲解了数据的增、删、改、查操作,以及查询语句的条件、聚合函数、分组、排序和分页等知识点。
1048 56
【MySQL基础篇】全面学习总结SQL语法、DataGrip安装教程
|
关系型数据库 MySQL Java
Django学习二:配置mysql,创建model实例,自动创建数据库表,对mysql数据库表已经创建好的进行直接操作和实验。
这篇文章是关于如何使用Django框架配置MySQL数据库,创建模型实例,并自动或手动创建数据库表,以及对这些表进行操作的详细教程。
531 0
Django学习二:配置mysql,创建model实例,自动创建数据库表,对mysql数据库表已经创建好的进行直接操作和实验。
|
Java 关系型数据库 MySQL
springboot学习五:springboot整合Mybatis 连接 mysql数据库
这篇文章是关于如何使用Spring Boot整合MyBatis来连接MySQL数据库,并进行基本的增删改查操作的教程。
2763 0
springboot学习五:springboot整合Mybatis 连接 mysql数据库

推荐镜像

更多