一 介绍

存储引擎决定了表的类型,而表内存放的数据也要有不同的类型,每种数据类型都有自己的宽度,但宽度是可选的

详细参考:

  • http://www.runoob.com/mysql/mysql-data-types.html
  • http://dev.mysql.com/doc/refman/5.7/en/data-type-overview.html

mysql常用数据类型概览

  1. #1. 数字:
  2. 整型:tinyinit int bigint
  3. 小数:
  4. float :在位数比较短的情况下不精准
  5. double :在位数比较长的情况下不精准
  6. 0.000001230123123123
  7. 存成:0.000001230000
  8.  
  9. decimal:(如果用小数,则用推荐使用decimal
  10. 精准
  11. 内部原理是以字符串形式去存
  12.  
  13. #2. 字符串:
  14. char10):简单粗暴,浪费空间,存取速度快
  15. root存成root000000
  16. varchar:精准,节省空间,存取速度慢
  17.  
  18. sql优化:创建表时,定长的类型往前放,变长的往后放
  19. 比如性别 比如地址或描述信息
  20.  
  21. >255个字符,超了就把文件路径存放到数据库中。
  22. 比如图片,视频等找一个文件服务器,数据库中只存路径或url
  23.  
  24. #3. 时间类型:
  25. 最常用:datetime
  26.  
  27. #4. 枚举类型与集合类型

二 数值类型

1、整数类型

整数类型:TINYINT SMALLINT MEDIUMINT INT BIGINT

作用:存储年龄,等级,id,各种号码等

  1. ========================================
  2. tinyint[(m)] [unsigned] [zerofill]
  3.  
  4. 小整数,数据类型用于保存一些范围的整数数值范围:
  5. 有符号:
  6. -128 127
  7. 无符号:
  8. 0 255
  9.  
  10. PS MySQL中无布尔值,使用tinyint(1)构造。
  11.  
  12. ========================================
  13. int[(m)][unsigned][zerofill]
  14.  
  15. 整数,数据类型用于保存一些范围的整数数值范围:
  16. 有符号:
  17. -2147483648 2147483647
  18. 无符号:
  19. 0 4294967295
  20.  
  21. ========================================
  22. bigint[(m)][unsigned][zerofill]
  23. 大整数,数据类型用于保存一些范围的整数数值范围:
  24. 有符号:
  25. -9223372036854775808 9223372036854775807
  26. 无符号:
  27. 0 18446744073709551615
  1. =========有符号和无符号tinyint==========
  2. #tinyint默认为有符号
  3. MariaDB [db1]> create table t1(x tinyint); #默认为有符号,即数字前有正负号
  4. MariaDB [db1]> desc t1;
  5. MariaDB [db1]> insert into t1 values
  6. -> (-129),
  7. -> (-128),
  8. -> (127),
  9. -> (128);
  10. MariaDB [db1]> select * from t1;
  11. +------+
  12. | x |
  13. +------+
  14. | -128 | #-129存成了-128
  15. | -128 | #有符号,最小值为-128
  16. | 127 | #有符号,最大值127
  17. | 127 | #128存成了127
  18. +------+
  19.  
  20. #设置无符号tinyint
  21. MariaDB [db1]> create table t2(x tinyint unsigned);
  22. MariaDB [db1]> insert into t2 values
  23. -> (-1),
  24. -> (0),
  25. -> (255),
  26. -> (256);
  27. MariaDB [db1]> select * from t2;
  28. +------+
  29. | x |
  30. +------+
  31. | 0 | -1存成了0
  32. | 0 | #无符号,最小值为0
  33. | 255 | #无符号,最大值为255
  34. | 255 | #256存成了255
  35. +------+
  36.  
  37. ============有符号和无符号int=============
  38. #int默认为有符号
  39. MariaDB [db1]> create table t3(x int); #默认为有符号整数
  40. MariaDB [db1]> insert into t3 values
  41. -> (-2147483649),
  42. -> (-2147483648),
  43. -> (2147483647),
  44. -> (2147483648);
  45. MariaDB [db1]> select * from t3;
  46. +-------------+
  47. | x |
  48. +-------------+
  49. | -2147483648 | #-2147483649存成了-2147483648
  50. | -2147483648 | #有符号,最小值为-2147483648
  51. | 2147483647 | #有符号,最大值为2147483647
  52. | 2147483647 | #2147483648存成了2147483647
  53. +-------------+
  54.  
  55. #设置无符号int
  56. MariaDB [db1]> create table t4(x int unsigned);
  57. MariaDB [db1]> insert into t4 values
  58. -> (-1),
  59. -> (0),
  60. -> (4294967295),
  61. -> (4294967296);
  62. MariaDB [db1]> select * from t4;
  63. +------------+
  64. | x |
  65. +------------+
  66. | 0 | #-1存成了0
  67. | 0 | #无符号,最小值为0
  68. | 4294967295 | #无符号,最大值为4294967295
  69. | 4294967295 | #4294967296存成了4294967295
  70. +------------+
  71.  
  72. ==============有符号和无符号bigint=============
  73. MariaDB [db1]> create table t6(x bigint);
  74. MariaDB [db1]> insert into t5 values
  75. -> (-9223372036854775809),
  76. -> (-9223372036854775808),
  77. -> (9223372036854775807),
  78. -> (9223372036854775808);
  79.  
  80. MariaDB [db1]> select * from t5;
  81. +----------------------+
  82. | x |
  83. +----------------------+
  84. | -9223372036854775808 |
  85. | -9223372036854775808 |
  86. | 9223372036854775807 |
  87. | 9223372036854775807 |
  88. +----------------------+
  89.  
  90. MariaDB [db1]> create table t6(x bigint unsigned);
  91. MariaDB [db1]> insert into t6 values
  92. -> (-1),
  93. -> (0),
  94. -> (18446744073709551615),
  95. -> (18446744073709551616);
  96.  
  97. MariaDB [db1]> select * from t6;
  98. +----------------------+
  99. | x |
  100. +----------------------+
  101. | 0 |
  102. | 0 |
  103. | 18446744073709551615 |
  104. | 18446744073709551615 |
  105. +----------------------+
  106.  
  107. ======用zerofill测试整数类型的显示宽度=============
  108. MariaDB [db1]> create table t7(x int(3) zerofill);
  109. MariaDB [db1]> insert into t7 values
  110. -> (1),
  111. -> (11),
  112. -> (111),
  113. -> (1111);
  114. MariaDB [db1]> select * from t7;
  115. +------+
  116. | x |
  117. +------+
  118. | 001 |
  119. | 011 |
  120. | 111 |
  121. | 1111 | #超过宽度限制仍然可以存
  122. +------+

验证

注意:为该类型指定宽度时,仅仅只是指定查询结果的显示宽度,与存储范围无关,存储范围如下

其实我们完全没必要为整数类型指定显示宽度,使用默认的就可以了

默认的显示宽度,都是在最大值的基础上加1

int的存储宽度是4个Bytes,即32个bit,即2**32

无符号最大值为:4294967296-1

有符号最大值:2147483648-1

有符号和无符号的最大数字需要的显示宽度均为10,而针对有符号的最小值则需要11位才能显示完全,所以int类型默认的显示宽度为11是非常合理的

最后:整形类型,其实没有必要指定显示宽度,使用默认的就ok

2、浮点型

定点数类型  DEC等同于DECIMAL  

浮点类型:FLOAT DOUBLE

作用:存储薪资、身高、体重、体质参数等

  1. ======================================
  2. #FLOAT[(M,D)] [UNSIGNED] [ZEROFILL]
  3.  
  4. 定义:
  5. 单精度浮点数(非准确小数值),m是数字总个数,d是小数点后个数。m最大值为255d最大值为30
  6.  
  7. 有符号:
  8. -3.402823466E+38 to -1.175494351E-38,
  9. 1.175494351E-38 to 3.402823466E+38
  10. 无符号:
  11. 1.175494351E-38 to 3.402823466E+38
  12.  
  13. 精确度:
  14. **** 随着小数的增多,精度变得不准确 ****
  15.  
  16. ======================================
  17. #DOUBLE[(M,D)] [UNSIGNED] [ZEROFILL]
  18.  
  19. 定义:
  20. 双精度浮点数(非准确小数值),m是数字总个数,d是小数点后个数。m最大值为255d最大值为30
  21.  
  22. 有符号:
  23. -1.7976931348623157E+308 to -2.2250738585072014E-308
  24. 2.2250738585072014E-308 to 1.7976931348623157E+308
  25.  
  26. 无符号:
  27. 2.2250738585072014E-308 to 1.7976931348623157E+308
  28.  
  29. 精确度:
  30. ****随着小数的增多,精度比float要高,但也会变得不准确 ****
  31.  
  32. ======================================
  33. decimal[(m[,d])] [unsigned] [zerofill]
  34.  
  35. 定义:
  36. 准确的小数值,m是数字总个数(负号不算),d是小数点后个数。 m最大值为65d最大值为30
  37.  
  38. 精确度:
  39. **** 随着小数的增多,精度始终准确 ****
  40. 对于精确数值计算时需要用此类型
  41. decaimal能够存储精确值的原因在于其内部按照字符串存储。
  1. mysql> create table t1(x float(256,31));
  2. ERROR 1425 (42000): Too big scale 31 specified for column 'x'. Maximum is 30.
  3. mysql> create table t1(x float(256,30));
  4. ERROR 1439 (42000): Display width out of range for column 'x' (max = 255)
  5. mysql> create table t1(x float(255,30)); #建表成功
  6. Query OK, 0 rows affected (0.02 sec)
  7.  
  8. mysql> create table t2(x double(255,30)); #建表成功
  9. Query OK, 0 rows affected (0.02 sec)
  10.  
  11. mysql> create table t3(x decimal(66,31));
  12. ERROR 1425 (42000): Too big scale 31 specified for column 'x'. Maximum is 30.
  13. mysql> create table t3(x decimal(66,30));
  14. ERROR 1426 (42000): Too-big precision 66 specified for 'x'. Maximum is 65.
  15. mysql> create table t3(x decimal(65,30)); #建表成功
  16. Query OK, 0 rows affected (0.02 sec)
  17.  
  18. mysql> show tables;
  19. +---------------+
  20. | Tables_in_db1 |
  21. +---------------+
  22. | t1 |
  23. | t2 |
  24. | t3 |
  25. +---------------+
  26. 3 rows in set (0.00 sec)
  27.  
  28. mysql> insert into t1 values(1.1111111111111111111111111111111); #小数点后31个1
  29. Query OK, 1 row affected (0.01 sec)
  30.  
  31. mysql> insert into t2 values(1.1111111111111111111111111111111);
  32. Query OK, 1 row affected (0.00 sec)
  33.  
  34. mysql> insert into t3 values(1.1111111111111111111111111111111);
  35. Query OK, 1 row affected, 1 warning (0.01 sec)
  36.  
  37. mysql> select * from t1; #随着小数的增多,精度开始不准确
  38. +----------------------------------+
  39. | x |
  40. +----------------------------------+
  41. | 1.111111164093017600000000000000 |
  42. +----------------------------------+
  43. 1 row in set (0.00 sec)
  44.  
  45. mysql> select * from t2; #精度比float要准确点,但随着小数的增多,同样变得不准确
  46. +----------------------------------+
  47. | x |
  48. +----------------------------------+
  49. | 1.111111111111111200000000000000 |
  50. +----------------------------------+
  51. 1 row in set (0.00 sec)
  52.  
  53. mysql> select * from t3; #精度始终准确,d为30,于是只留了30位小数
  54. +----------------------------------+
  55. | x |
  56. +----------------------------------+
  57. | 1.111111111111111111111111111111 |
  58. +----------------------------------+
  59. 1 row in set (0.00 sec)

验证

3、位类型(了解)

BIT(M)可以用来存放多位二进制数,M范围从1~64,如果不写默认为1位。
注意:对于位字段需要使用函数读取
    bin()显示为二进制
    hex()显示为十六进制

  1. MariaDB [db1]> create table t9(id bit);
  2. MariaDB [db1]> desc t9; #bit默认宽度为1
  3. +-------+--------+------+-----+---------+-------+
  4. | Field | Type | Null | Key | Default | Extra |
  5. +-------+--------+------+-----+---------+-------+
  6. | id | bit(1) | YES | | NULL | |
  7. +-------+--------+------+-----+---------+-------+
  8.  
  9. MariaDB [db1]> insert into t9 values(8);
  10. MariaDB [db1]> select * from t9; #直接查看是无法显示二进制位的
  11. +------+
  12. | id |
  13. +------+
  14. | |
  15. +------+
  16. MariaDB [db1]> select bin(id),hex(id) from t9; #需要转换才能看到
  17. +---------+---------+
  18. | bin(id) | hex(id) |
  19. +---------+---------+
  20. | 1 | 1 |
  21. +---------+---------+
  22.  
  23. MariaDB [db1]> alter table t9 modify id bit(5);
  24. MariaDB [db1]> insert into t9 values(8);
  25. MariaDB [db1]> select bin(id),hex(id) from t9;
  26. +---------+---------+
  27. | bin(id) | hex(id) |
  28. +---------+---------+
  29. | 1 | 1 |
  30. | 1000 | 8 |
  31. +---------+---------+

验证

三 日期类型

DATE TIME DATETIME TIMESTAMP YEAR

作用:存储用户注册时间,文章发布时间,员工入职时间,出生时间,过期时间等

  1. YEAR
  2. YYYY1901/2155
  3.  
  4. DATE
  5. YYYY-MM-DD1000-01-01/9999-12-31
  6.  
  7. TIME
  8. HH:MM:SS'-838:59:59'/'838:59:59'
  9.  
  10. DATETIME
  11.  
  12. YYYY-MM-DD HH:MM:SS1000-01-01 00:00:00/9999-12-31 23:59:59 Y
  13.  
  14. TIMESTAMP
  15.  
  16. YYYYMMDD HHMMSS1970-01-01 00:00:00/2037 年某时)
  1. ============year===========
  2. MariaDB [db1]> create table t10(born_year year); #无论year指定何种宽度,最后都默认是year(4)
  3. MariaDB [db1]> insert into t10 values
  4. -> (1900),
  5. -> (1901),
  6. -> (2155),
  7. -> (2156);
  8. MariaDB [db1]> select * from t10;
  9. +-----------+
  10. | born_year |
  11. +-----------+
  12. | 0000 |
  13. | 1901 |
  14. | 2155 |
  15. | 0000 |
  16. +-----------+
  17.  
  18. ============date,time,datetime===========
  19. MariaDB [db1]> create table t11(d date,t time,dt datetime);
  20. MariaDB [db1]> desc t11;
  21. +-------+----------+------+-----+---------+-------+
  22. | Field | Type | Null | Key | Default | Extra |
  23. +-------+----------+------+-----+---------+-------+
  24. | d | date | YES | | NULL | |
  25. | t | time | YES | | NULL | |
  26. | dt | datetime | YES | | NULL | |
  27. +-------+----------+------+-----+---------+-------+
  28.  
  29. MariaDB [db1]> insert into t11 values(now(),now(),now());
  30. MariaDB [db1]> select * from t11;
  31. +------------+----------+---------------------+
  32. | d | t | dt |
  33. +------------+----------+---------------------+
  34. | 2017-07-25 | 16:26:54 | 2017-07-25 16:26:54 |
  35. +------------+----------+---------------------+
  36.  
  37. ============timestamp===========
  38. MariaDB [db1]> create table t12(time timestamp);
  39. MariaDB [db1]> insert into t12 values();
  40. MariaDB [db1]> insert into t12 values(null);
  41. MariaDB [db1]> select * from t12;
  42. +---------------------+
  43. | time |
  44. +---------------------+
  45. | 2017-07-25 16:29:17 |
  46. | 2017-07-25 16:30:01 |
  47. +---------------------+
  48.  
  49. ============注意啦,注意啦,注意啦===========
  50. 1. 单独插入时间时,需要以字符串的形式,按照对应的格式插入
  51. 2. 插入年份时,尽量使用4位值
  52. 3. 插入两位年份时,<=69,以20开头,比如50, 结果2050
  53. >=70,以19开头,比如71,结果1971
  54. MariaDB [db1]> create table t12(y year);
  55. MariaDB [db1]> insert into t12 values
  56. -> (50),
  57. -> (71);
  58. MariaDB [db1]> select * from t12;
  59. +------+
  60. | y |
  61. +------+
  62. | 2050 |
  63. | 1971 |
  64. +------+
  65.  
  66. ============综合练习===========
  67. MariaDB [db1]> create table student(
  68. -> id int,
  69. -> name varchar(20),
  70. -> born_year year,
  71. -> birth date,
  72. -> class_time time,
  73. -> reg_time datetime);
  74.  
  75. MariaDB [db1]> insert into student values
  76. -> (1,'alex',"","1995-11-11","11:11:11","2017-11-11 11:11:11"),
  77. -> (2,'egon',"","1997-12-12","12:12:12","2017-12-12 12:12:12"),
  78. -> (3,'wsb',"","1998-01-01","13:13:13","2017-01-01 13:13:13");
  79.  
  80. MariaDB [db1]> select * from student;
  81. +------+------+-----------+------------+------------+---------------------+
  82. | id | name | born_year | birth | class_time | reg_time |
  83. +------+------+-----------+------------+------------+---------------------+
  84. | 1 | alex | 1995 | 1995-11-11 | 11:11:11 | 2017-11-11 11:11:11 |
  85. | 2 | egon | 1997 | 1997-12-12 | 12:12:12 | 2017-12-12 12:12:12 |
  86. | 3 | wsb | 1998 | 1998-01-01 | 13:13:13 | 2017-01-01 13:13:13 |
  87. +------+------+-----------+------------+------------+---------------------+

验证

  1. 在实际应用的很多场景中,MySQL的这两种日期类型都能够满足我们的需要,存储精度都为秒,但在某些情况下,会展现出他们各自的优劣。下面就来总结一下两种日期类型的区别。
  2.  
  3. 1.DATETIME的日期范围是1001——9999年,TIMESTAMP的时间范围是1970——2038年。
  4.  
  5. 2.DATETIME存储时间与时区无关,TIMESTAMP存储时间与时区有关,显示的值也依赖于时区。在mysql服务器,操作系统以及客户端连接都有时区的设置。
  6.  
  7. 3.DATETIME使用8字节的存储空间,TIMESTAMP的存储空间为4字节。因此,TIMESTAMPDATETIME的空间利用率更高。
  8.  
  9. 4.DATETIME的默认值为nullTIMESTAMP的字段默认不为空(not null),默认值为当前时间(CURRENT_TIMESTAMP),如果不做特殊处理,并且update语句中没有指定该列的更新值,则默认更新为当前时间。

datetime与timestamp的区别

四 字符串类型

  1. #官网:https://dev.mysql.com/doc/refman/5.7/en/char.html
  2. #注意:char和varchar括号内的参数指的都是字符的长度
  3.  
  4. #char类型:定长,简单粗暴,浪费空间,存取速度快
  5. 字符长度范围:0-255(一个中文是一个字符,是utf8编码的3个字节)
  6. 存储:
  7. 存储char类型的值时,会往右填充空格来满足长度
  8. 例如:指定长度为10,存>10个字符则报错,存<10个字符则用空格填充直到凑够10个字符存储
  9.  
  10. 检索:
  11. 在检索或者说查询时,查出的结果会自动删除尾部的空格,除非我们打开pad_char_to_full_length SQL模式(SET sql_mode = 'PAD_CHAR_TO_FULL_LENGTH';)
  12.  
  13. #varchar类型:变长,精准,节省空间,存取速度慢
  14. 字符长度范围:0-65535(如果大于21845会提示用其他类型 mysql行最大限制为65535字节,字符编码为utf-8https://dev.mysql.com/doc/refman/5.7/en/column-count-limit.html)
  15. 存储:
  16. varchar类型存储数据的真实内容,不会用空格填充,如果'ab ',尾部的空格也会被存起来
  17. 强调:varchar类型会在真实数据前加1-2Bytes的前缀,该前缀用来表示真实数据的bytes字节数(1-2Bytes最大表示65535个数字,正好符合mysqlrow的最大字节限制,即已经足够使用)
  18. 如果真实的数据<255bytes则需要1Bytes的前缀(1Bytes=8bit 2**8最大表示的数字为255
  19. 如果真实的数据>255bytes则需要2Bytes的前缀(2Bytes=16bit 2**16最大表示的数字为65535
  20.  
  21. 检索:
  22. 尾部有空格会保存下来,在检索或者说查询时,也会正常显示包含空格在内的内容
  1. #官网:https://dev.mysql.com/doc/refman/5.7/en/char.html
  2. CHAR VARCHAR 是最常使用的两种字符串类型。
  3. 一般来说
  4. CHAR(N)用来保存固定长度的字符串,对于 CHAR 类型,N 的范围 0 ~ 255
  5. VARCHAR(N)用来保存变长字符类型,对于 VARCHAR 类型,N 的范围为 0 ~ 65 535
  6. CHAR(N)和 VARCHAR(N) 中的 N 都代表字符长度,而非字节长度。
  7. ps:对于 MySQL 4.1 之前的版本,如 MySQL 3.23 MySQL 4.0,CHAR(N)和 VARCHAR (N)中的 N 代表字节长度。
  8.  
  9. #CHAR类型
  10. 对于 CHAR 类型的字符串,MySQL 数据库会自动对存储列的右边进行填充(Right Padded)操作,直到字符串达到指定的长度 N。而在读取该列时,MySQL 数据库会自动将 填充的字符删除。有一种情况例外,那就是显式地将 SQL_MODE 设置为 PAD_CHAR_TO_ FULL_LENGTH,例如:
  11. mysql> CREATE TABLE t ( a CHAR(10));
  12. Query OK, 0 rows affected (0.03 sec)
  13. mysql> INSERT INTO t SELECT 'abc';
  14. Query OK, 1 row affected (0.03 sec)
  15. Records: 1 Duplicates: 0 Warnings: 0
  16. mysql> SELECT a,HEX(a),LENGTH(a) FROM t\G;
  17. *************************** 1. row ***************************
  18. a: abc
  19. HEX(a): 616263
  20. LENGTH (a): 3
  21. 1 row in set (0.00 sec)
  22. mysql> SET SQL_MODE='PAD_CHAR_TO_FULL_LENGTH';
  23. Query OK, 0 rows affected (0.00 sec)
  24. mysql> SELECT a,HEX(a),LENGTH(a) FROM t\G;
  25. *************************** 1. row ***************************
  26. a: abc
  27. HEX(a): 61626320202020202020
  28. LENGTH (a): 10
  29. 1 row in set (0.00 sec)
  30.  
  31. 在上述这个例子中,先创建了一张表 t,a 列的类型为 CHAR(10)。然后通过 INSERT语句插入值“abc”,因为 a 列的类型为 CHAR 型,所以会自动在后面填充空字符串,使其长 度为 10。接下来在通过 SELECT 语句取出数据时会将 a 列右填充的空字符移除,从而得到 值“abc”。通过 LENGTH 函数看到 a 列的字符长度为 3 而非 10
  32. 接着我们将 SQL_MODE 显式地设置为 PAD_CHAR_TO_FULL_LENGTH。这时再通过 SELECT 语句进行查询时,得到的结果是“abc ”,abc 右边有 7 个填充字符 0x20,并通 HEX 函数得到了验证。这次 LENGTH 函数返回的长度为 10。需要注意的是,LENGTH 函数返回的是字节长度,而不是字符长度。对于多字节字符集,CHAR(N)长度的列最多 可占用的字节数为该字符集单字符最大占用字节数 *N。例如,对于 utf8 下,CHAR(10)最 多可能占用 30 个字节。通过对多字节字符串使用 CHAR_LENGTH 函数和 LENGTH 函数, 可以发现两者的不同,示例如下:
  33. mysql> SET NAMES gbk;
  34. Query OK, 0 rows affected (0.03 sec)
  35. mysql> SELECT @a:='MySQL 技术内幕 '; Query OK, 0 rows affected (0.03 sec)
  36. mysql> SELECT @a,HEX(@a),LENGTH(@a),CHAR_LENGTH(@a)\G; ***************************** 1. row **************************** a: MySQL 技术内幕
  37. HEX(a): 4D7953514CBCBCCAF5C4DAC4BB
  38. LENGTH (a): 13
  39. CHAR_LENGTH(a): 9
  40. 1 row in set (0.00 sec)
  41.  
  42. @ a g b k , M y S Q L ”, 0x4D7953514CBCBCCAF5C4DAC4BB,LENGTH 函数返回 13,即该字符串占用 13 字节, 因为 gbk 字符集中的中文字符占用两个字节,因此一共占用 13 字节。CHAR_LENGTH 函数 返回 9,很显然该字符长度为 9
  43.  
  44. #VARCHAR类型
  45. VARCHAR 类型存储变长字段的字符类型,与 CHAR 类型不同的是,其存储时需要在 前缀长度列表加上实际存储的字符,该字符占用 1 ~ 2 字节的空间。当存储的字符串长度小 255 字节时,其需要 1 字节的空间,当大于 255 字节时,需要 2 字节的空间。所以,对 于单字节的 latin1 来说,CHAR(10)和 VARCHAR(10)最大占用的存储空间是不同的, CHAR(10)占用 10 个字节这是毫无疑问的,而 VARCHAR(10)的最大占用空间数是 11 字节,因为其需要 1 字节来存放字符长度。
  46. -------------------------------------------------
  47. 注意 对于有些多字节的字符集类型,其 CHAR VARCHAR 在存储方法上是一样的,同样 需要为长度列表加上字符串的值。对于 GBK UTF-8 这些字符类型,其有些字符是以 1 字节 存放的,有些字符是按 2 3 字节存放的,因此同样需要 1 ~ 2 字节的空间来存储字符的长 度。
  48. -------------------------------------------------
  49. 虽然 CHAR VARCHAR 的存储方式不太相同,但是对于两个字符串的比较,都只比 较其值,忽略 CHAR 值存在的右填充,即使将 SQL _MODE 设置为 PAD_CHAR_TO_FULL_ LENGTH 也一样,例如:
  50. mysql> CREATE TABLE t ( a CHAR(10), b VARCHAR(10));
  51. Query OK, 0 rows affected (0.01 sec)
  52. mysql> INSERT INTO t SELECT 'a','a';
  53. Query OK, 1 row affected (0.00 sec)
  54. Records: 1 Duplicates: 0 Warnings: 0
  55. mysql> SELECT a=b FROM t\G;
  56. *************************** 1. row ***************************
  57. a=b: 1
  58. 1 row in set (0.00 sec)
  59. mysql> SET SQL_MODE='PAD_CHAR_TO_FULL_LENGTH';
  60. Query OK, 0 rows affected (0.00 sec)
  61. mysql> SELECT a=b FROM t\G;
  62. *************************** 1. row ***************************
  63. a=b: 1
  64. 1 row in set (0.00 sec)

官网详解

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
  1. 测试前了解两个函数
  2. length:查看字节数
  3. char_length:查看字符数

1. char填充空格来满足固定长度,但是在查询时却会很不要脸地删除尾部的空格(装作自己好像没有浪费过空间一样),然后修改sql_mode让其现出原形

  1. mysql> create table t1(x char(5),y varchar(5));
  2. Query OK, 0 rows affected (0.26 sec)
  3.  
  4. #char存5个字符,而varchar存4个字符
  5. mysql> insert into t1 values('你瞅啥 ','你瞅啥 ');
  6. Query OK, 1 row affected (0.05 sec)
  7.  
  8. mysql> SET sql_mode='';
  9. Query OK, 0 rows affected, 1 warning (0.00 sec)
  10.  
  11. #在检索时char很不要脸地将自己浪费的2个字符给删掉了,装的好像自己没浪费过空间一样,而varchar很老实,存了多少,就显示多少
  12. mysql> select x,char_length(x),y,char_length(y) from t1;
  13. +-----------+----------------+------------+----------------+
  14. | x | char_length(x) | y | char_length(y) |
  15. +-----------+----------------+------------+----------------+
  16. | 你瞅啥 | 3 | 你瞅啥 | 4 |
  17. +-----------+----------------+------------+----------------+
  18. 1 row in set (0.00 sec)
  19.  
  20. #略施小计,让char现出原形
  21. mysql> SET sql_mode = 'PAD_CHAR_TO_FULL_LENGTH';
  22. Query OK, 0 rows affected (0.00 sec)
  23.  
  24. #这下子char原形毕露了......
  25. mysql> select x,char_length(x),y,char_length(y) from t1;
  26. +-------------+----------------+------------+----------------+
  27. | x | char_length(x) | y | char_length(y) |
  28. +-------------+----------------+------------+----------------+
  29. | 你瞅啥 | 5 | 你瞅啥 | 4 |
  30. +-------------+----------------+------------+----------------+
  31. 1 row in set (0.00 sec)
  32.  
  33. #char类型:3个中文字符+2个空格=11Bytes
  34. #varchar类型:3个中文字符+1个空格=10Bytes
  35. mysql> select x,length(x),y,length(y) from t1;
  36. +-------------+-----------+------------+-----------+
  37. | x | length(x) | y | length(y) |
  38. +-------------+-----------+------------+-----------+
  39. | 你瞅啥 | 11 | 你瞅啥 | 10 |
  40. +-------------+-----------+------------+-----------+
  41. 1 row in set (0.00 sec)
  1. mysql> select concat('数据: ',x,'长度: ',char_length(x)),concat(y,char_length(y)
  2. ) from t1;
  3. +------------------------------------------------+--------------------------+
  4. | concat('数据: ',x,'长度: ',char_length(x)) | concat(y,char_length(y)) |
  5. +------------------------------------------------+--------------------------+
  6. | 数据: 你瞅啥 长度: 5 | 你瞅啥 4 |
  7. +------------------------------------------------+--------------------------+
  8. 1 row in set (0.00 sec)

了解concat

2. 虽然 CHAR 和 VARCHAR 的存储方式不太相同,但是对于两个字符串的比较,都只比 较其值,忽略 CHAR 值存在的右填充,即使将 SQL _MODE 设置为 PAD_CHAR_TO_FULL_ LENGTH 也一样,,但这不适用于like

  1. Values in CHAR and VARCHAR columns are sorted and compared according to the character set collation assigned to the column.
  2.  
  3. All MySQL collations are of type PAD SPACE. This means that all CHAR, VARCHAR, and TEXT values are compared without regard to any trailing spaces. Comparison in this context does not include the LIKE pattern-matching operator, for which trailing spaces are significant. For example:
  4.  
  5. mysql> CREATE TABLE names (myname CHAR(10));
  6. Query OK, 0 rows affected (0.03 sec)
  7.  
  8. mysql> INSERT INTO names VALUES ('Monty');
  9. Query OK, 1 row affected (0.00 sec)
  10.  
  11. mysql> SELECT myname = 'Monty', myname = 'Monty ' FROM names;
  12. +------------------+--------------------+
  13. | myname = 'Monty' | myname = 'Monty ' |
  14. +------------------+--------------------+
  15. | 1 | 1 |
  16. +------------------+--------------------+
  17. 1 row in set (0.00 sec)
  18.  
  19. mysql> SELECT myname LIKE 'Monty', myname LIKE 'Monty ' FROM names;
  20. +---------------------+-----------------------+
  21. | myname LIKE 'Monty' | myname LIKE 'Monty ' |
  22. +---------------------+-----------------------+
  23. | 1 | 0 |
  24. +---------------------+-----------------------+
  25. 1 row in set (0.00 sec)

3. 总结

  1. #常用字符串系列:char与varchar
  2. 注:虽然varchar使用起来较为灵活,但是从整个系统的性能角度来说,char数据类型的处理速度更快,有时甚至可以超出varchar处理速度的50%。因此,用户在设计数据库时应当综合考虑各方面的因素,以求达到最佳的平衡
  3.  
  4. #其他字符串系列(效率:char>varchar>text)
  5. TEXT系列 TINYTEXT TEXT MEDIUMTEXT LONGTEXT
  6. BLOB 系列 TINYBLOB BLOB MEDIUMBLOB LONGBLOB
  7. BINARY系列 BINARY VARBINARY
  8.  
  9. texttext数据类型用于保存变长的大字符串,可以组多到65535 (2**16 1)个字符。
  10. mediumtextA TEXT column with a maximum length of 16,777,215 (2**24 1) characters.
  11. longtextA TEXT column with a maximum length of 4,294,967,295 or 4GB (2**32 1) characters.

五 枚举类型与集合类型

字段的值只能在给定范围中选择,如单选框,多选框
enum 单选 只能在给定的范围内选一个值,如性别 sex 男male/女female
set 多选 在给定的范围内可以选择一个或一个以上的值(爱好1,爱好2,爱好3...)

  1. 枚举类型(enum
  2. An ENUM column can have a maximum of 65,535 distinct elements. (The practical limit is less than 3000.)
  3. 示例:
  4. CREATE TABLE shirts (
  5. name VARCHAR(40),
  6. size ENUM('x-small', 'small', 'medium', 'large', 'x-large')
  7. );
  8. INSERT INTO shirts (name, size) VALUES ('dress shirt','large'), ('t-shirt','medium'),('polo shirt','small');
  9.  
  10. 集合类型(set
  11. A SET column can have a maximum of 64 distinct members.
  12. 示例:
  13. CREATE TABLE myset (col SET('a', 'b', 'c', 'd'));
  14. INSERT INTO myset (col) VALUES ('a,d'), ('d,a'), ('a,d,a'), ('a,d,d'), ('d,a,d');
  1. MariaDB [db1]> create table consumer(
  2. -> name varchar(50),
  3. -> sex enum('male','female'),
  4. -> level enum('vip1','vip2','vip3','vip4','vip5'), #在指定范围内,多选一
  5. -> hobby set('play','music','read','study') #在指定范围内,多选多
  6. -> );
  7.  
  8. MariaDB [db1]> insert into consumer values
  9. -> ('egon','male','vip5','read,study'),
  10. -> ('alex','female','vip1','girl');
  11.  
  12. MariaDB [db1]> select * from consumer;
  13. +------+--------+-------+------------+
  14. | name | sex | level | hobby |
  15. +------+--------+-------+------------+
  16. | egon | male | vip5 | read,study |
  17. | alex | female | vip1 | |
  18. +------+--------+-------+------------+

验证

 
 
 

Mysql(三)-2:数据类型的更多相关文章

  1. mysql三-2:数据类型

    一 介绍 存储引擎决定了表的类型,而表内存放的数据也要有不同的类型,每种数据类型都有自己的宽度,但宽度是可选的 详细参考: http://www.runoob.com/mysql/mysql-data ...

  2. python、mysql三-2:数据类型

    一 介绍 存储引擎决定了表的类型,而表内存放的数据也要有不同的类型,每种数据类型都有自己的宽度,但宽度是可选的 详细参考: http://www.runoob.com/mysql/mysql-data ...

  3. 一、初识MySQL数据库 二、搭建MySQL数据库(重点) 三、使用MySQL数据库 四、认识MySQL数据库的数据类型 五、操作MySQL数据库的数据(重点)

    一.初识MySQL数据库 ###<1>数据库概述     1. 数据库         长期存储在计算机内的,由组织的可共享的数据集合         存储数据的仓库         文件 ...

  4. Mysql的三种数据类型

    Mysql的三种数据类型 1.数值类型 2.日期和时间类型 3.字符串类型 00x1 [数值类型] 00x2 [日期和时间类型] 00x3 [字符串类型]

  5. MySQL 的数值数据类型

    MySQL 的数值数据类型可以大致划分为两个类别,一个是整数,另一个是浮点数或小数.许多不同的子类型对这些类别中的每一个都是可用的,每个子类型支持不同大小的数据,并且 MySQL 允许我们指定数值字段 ...

  6. 高性能MySql进化论(一):数据类型的优化_上

    在数据库的性能调优的过程中会涉及到很多的知识,包括字段的属性设置是否合适,索引的建立是否恰当,表结构涉及是否合理,数据库/操作系统 的设置是否正确…..其中每个topic可能都是一个领域. 在我看来, ...

  7. MySQL中各种数据类型的长度及在开发中如何选择

    接触MySQL这个数据库大概快要两年了,不过由于没有特别深入系统的去学习,大多也是停留在一知半解的状态.今天在工作中刚好碰到了表设计的问题,顺便写篇博客,把MySQL中数据类型和字段类型选择这方面给弄 ...

  8. 存储引擎和表的操作(mysql中的数据类型、完整性约束)

    一.存储引擎 .概念 MySQL中的数据用各种不同的技术存储在文件(或者内存)中.这些技术中的每一种技术都使用不同的存储机制.索引技巧.锁定水平并且最终提供广泛的不同的功能和能力. 通过选择不同的技术 ...

  9. 数据库 -- mysql支持的数据类型

    mysql支持的数据类型 数值类型 MySQL支持所有标准SQL数值数据类型. 这些类型包括严格数值数据类型(INTEGER.SMALLINT.DECIMAL和NUMERIC),以及近似数值数据类型( ...

随机推荐

  1. SQLite学习手册(实例代码<一>)

    一.获取表的Schema信息:       1). 动态创建表.     2). 根据sqlite3提供的API,获取表字段的信息,如字段数量以及每个字段的类型.     3). 删除该表.     ...

  2. javabean内省

    何为JavaBean? JavaBean 是一种JAVA语言写成的可重用组件.为写成JavaBean,类必须是具体的和公共的,并且具有无参数的构造器.JavaBean 通过提供符合一致性设计模式的公共 ...

  3. 计算机改名引发的ORA

    近期上班时,由于开机时老是提示" 局域网出现计算机重名冲突",于是把计算机名字给改了,从PC2010081312zeo改为了CXBIKKKKKKK,结果第二天来的时候,用PL/SQ ...

  4. 使用keepAlive对上下拉刷新列表数据 和 滚动位置细节处理 - vue

    [前言] 使用vue处理项目中遇到列表页面时,之前项目中总会有一些细节问题处理得不太好,这里总结一下,以便优化以后的代码.如下: 1. 使用mint-ui中的LoadMore组件上下拉刷新时,有时无法 ...

  5. IO (二)

    1 字符流的缓冲区 缓冲区的出现提高了对数据的读写效率. 对应的类: BufferedWriter BufferedReader 缓冲区要结合流才能使用. 在流的基础上对流的功能进行了增强. 2 Bu ...

  6. 123 A. Prime Permutation

    链接 http://codeforces.com/contest/123/problem/A 题目 You are given a string s, consisting of small Lati ...

  7. mybatis支持oracle批量插入

    问题:mysql使用mybatis批量插入时,通过foreach标签,将每条记录按照逗号","连接即可. 但是,oracle不支持. oracle支持如下写法: <inser ...

  8. JAVA并发编程学习笔记------锁顺序死锁

    一.需求描述: 将资金从一个账户转移到另一个账户. 二.程序实现: (1)账户类: public class Account { private long account; public Accoun ...

  9. Django REST framework中的版本控制

    1.REST framework版本控制的流程分析 1.1 determine_version方法的执行流程 首先,请求到达REST framework的CBV,执行CBV中的dispatch方法再次 ...

  10. iOS-隐藏Navigationbar【导航栏无缝圆滑的隐藏】

    1.ViewController .m - (void)viewDidLoad { [super viewDidLoad]; self.title = @"隐藏导航栏"; UIBu ...