mysql更改字段类型 第01期:MySQL 数据类型的艺术

数据建模:字段类型-MySQL 数据类型的艺术

综上所述,日期这块类型的选择遵循以下原则:

1. 如果时间有可能超过时间戳范围,优先选择 datetime

2. 如果需要单独获取年份值,比如按照年来分区,按照年来检索等,最好在表中添加一个 year 类型来参与。

3. 如果需要单独获取日期或者时间,最好是单独存放,而不是简单的用 datetime 或者 timestamp。后面检索时,再加函数过滤,以免后期增加 SQL 编写带来额外消耗。

4. 如果有保存毫秒类似的需求,最好是用时间类型自己的特性,不要直接用字符类型来代替。MySQL 内部的类型转换对资源额外的消耗也是需要考虑的。

示例 5

建立表 t5,对这些可能需要的字段全部分离开,这样以后写 SQL 语句的时候就很容易了。

  1. mysql-(ytt/3305)->create table t5 (c1 date,c2 datetime(3),c3 timestamp(3),c4 time(3),c5 year);

  2. Query OK, 0 rows affected (1.01 sec)


  3. mysql-(ytt/3305)->set @a='2018-03-25 12:22:33.342';

  4. Query OK, 0 rows affected (0.00 sec)


  5. mysql-(ytt/3305)->insert into t5 values (date(@a), @a,@a,time(@a),year(@a));

  6. Query OK, 1 row affected (0.01 sec)


  7. mysql-(ytt/3305)->select * from t5;

  8. +------------+-------------------------+-------------------------+--------------+------+

  9. | c1 | c2 | c3 | c4 | c5 |

  10. +------------+-------------------------+-------------------------+--------------+------+

  11. | 2018-03-25 | 2018-03-25 12:22:33.342 | 2018-03-25 12:22:33.342 | 12:22:33.342 | 2018 |

  12. +------------+-------------------------+-------------------------+--------------+------+

当然了,这种情形占用额外的磁盘空间。如果想在易用性与空间占用量大这两点来折中,可以用 MySQL 的虚拟列来实时计算。比如假设 c5 字段不存在,想要得到 c5 的结果。

  1. mysql-(ytt/3305)->alter table t5 drop c5, add c5 year generated always as (year(c1)) virtual;

  2. Query OK, 1 row affected (2.46 sec)

  3. Records: 1 Duplicates: 0 Warnings: 0

五、二进制类型

binary 和 varbinary 对应了 char 和 varchar 的二进制存储,相关的特性都一样。不同的有以下几点:

示例 6

来看这个 binary 存取的简单示例,还是之前的变量 @a。

切记!这里要提前计算好 @a 占用的字节数,以防存储溢出。

  1. mysql-(ytt/3305)->set @a = "我是傻傻的小月亮!!!!";

  2. Query OK, 0 rows affected (0.00 sec)


  3. mysql-(ytt/3305)->create table t6 (c1 binary(28),c2 varbinary(28));

  4. Query OK, 0 rows affected (0.03 sec)


  5. mysql-(ytt/3305)->insert into t6 values (@a,@a);

  6. Query OK, 1 row affected (0.01 sec)


  7. mysql-(ytt/3305)->select * from t6;

  8. +------------------------------+------------------------------+

  9. | c1 | c2 |

  10. +------------------------------+------------------------------+

  11. | 我是傻傻的小月亮!!!! | 我是傻傻的小月亮!!!! |

  12. +------------------------------+------------------------------+

  13. 1 row in set (0.00 sec)

六、位类型

bit 为 MySQL 里存储比特位的类型,最大支持 64 比特位, 直接以二进制方式存储,一般用来存储状态类的信息。比如,性别,真假等。具有以下特性:

1. 对于 bit(8) 如果单纯存放 1 位,左边以 0 填充 00000001。

2. 查询时可以直接十进制来过滤数据。

3. 如果此字段加上索引,MySQL 不会自己做类型转换,只能用二进制来过滤。

示例 7

创建表 c1, 字段性别定义一个比特位。

  1. mysql-(ytt/3305)->create table c1(gender bit(1));

  2. Query OK, 0 rows affected (0.02 sec)

插入两条记录

  1. mysql-(ytt/3305)->insert into c1 values (b'0');

  2. Query OK, 1 row affected (0.01 sec)


  3. mysql-(ytt/3305)->insert into c1 values (b'1');

  4. Query OK, 1 row affected (0.00 sec)

检索数据,直接以十进制方式显示。

  1. mysql-(ytt/3305)->select gender+0 'f1' from c1;

  2. +------+

  3. | f1 |

  4. +------+

  5. | 0 |

  6. | 1 |

  7. +------+

  8. 2 rows in set (0.01 sec)

也可以用类型显示转换。

  1. mysql-(ytt/3305)->select cast(gender as unsigned) 'f1' from c1;

  2. +------+

  3. | f1 |

  4. +------+

  5. | 0 |

  6. | 1 |

  7. +------+

  8. 2 rows in set (0.00 sec)

过滤数据也一样,二进制或者直接十进制都行。

  1. mysql-(ytt/3305)->select conv(gender,16,10) as gender

  2.    -> from c1 where gender = b'1'; 

  3. +--------+

  4. | gender |

  5. +--------+

  6. | 1      |

  7. +--------+

  8. 1 row in set (0.00 sec)

  9.     

  10. mysql-(ytt/3305)->select conv(gender,16,10) as gender

  11.     -> from c1 where gender = '1';

  12. +--------+

  13. | gender |

  14. +--------+

  15. | 1      |

  16. +--------+

  17. 1 row in set (0.00 sec)

其实这样的场景,也可以定义为 char(0),这也是类似于 bit 非常优化的一种用法。

  1. mysql-(ytt/3305)->create table c2(gender char(0));

  2. Query OK, 0 rows affected (0.03 sec)

那现在我给表 c1 简单的造点测试数据。

  1. mysql-(ytt/3305)->select count(*) from c1;

  2. +----------+

  3. | count(*) |

  4. +----------+

  5. | 33554432 |

  6. +----------+

  7. 1 row in set (1.37 sec)

把 c1 的数据全部插入 c2。

  1. mysql-(ytt/3305)->insert into c2 select if(gender = 0,'',null) from c1;

  2. Query OK, 33554432 rows affected (2 min 18.80 sec)

  3. Records: 33554432 Duplicates: 0 Warnings: 0

两张表的磁盘占用差不多。

  1. root@ytt-pc:/var/lib/mysql/3305/ytt# ls -sihl

  2. 总用量 1.9G

  3. 4085684 933M -rw-r----- 1 mysql mysql 932M 12月 11 10:16 c1.ibd

  4. 4082686 917M -rw-r----- 1 mysql mysql 916M 12月 11 10:22 c2.ibd

检索方式稍微有些不同,不过效率也差不多。所以说,字符类型不愧为万能类型。

  1. mysql-(ytt/3305)->select count(*) from c1 where gender = 0;

  2. +----------+

  3. | count(*) |

  4. +----------+

  5. | 16772213 |

  6. +----------+

  7. 1 row in set (12.03 sec)


  8. mysql-(ytt/3305)->select count(*) from c2 where gender = '';

  9. +----------+

  10. | count(*) |

  11. +----------+

  12. | 16772213 |

  13. +----------+

  14. 1 row in set (12.53 sec)

七、枚举类型

枚举类型,也即 enum。适合提前规划好了所有已经知道的值,且未来最好不要加新值的情形。枚举类型有以下特性:

1. 最大占用 2 Byte。

2. 最大支持 65535 个不同元素。

3. MySQL 后台存储以下标的方式,也就是 tinyint 或者 smallint 的方式,下标从 1 开始。

4. 排序时按照下标排序,而不是按照里面元素的数据类型。所以这点要格外注意。

示例 8

创建表 t7。

  1. mysql-(ytt/3305)->create table t7(c1 enum('mysql','oracle','dble','postgresql','mongodb','redis','db2','sql server'));

  2. Query OK, 0 rows affected (0.03 sec)

用下标插入数据。

  1. mysql-(ytt/3305)->insert into t7 values (1);

  2. Query OK, 1 row affected (0.40 sec)


  3. mysql-(ytt/3305)->insert into t7 values (2);

  4. Query OK, 1 row affected (0.00 sec)

用真实元素插入数据。

  1. mysql-(ytt/3305)->insert into t7 values ('postgresql');

  2. Query OK, 1 row affected (0.01 sec)


  3. mysql-(ytt/3305)->insert into t7 values ('dble');

  4. Query OK, 1 row affected (0.01 sec)


  5. mysql-(ytt/3305)->insert into t7 values ('sql server');

查询结果出来,发现是按照之前定义的元素排序,也就是下标排序。

  1. mysql-(ytt/3305)->select * from t7 order by c1;

  2. +------------+

  3. | c1 |

  4. +------------+

  5. | mysql |

  6. | oracle |

  7. | dble |

  8. | postgresql |

  9. | sql server |

  10. +------------+

  11. 5 rows in set (0.00 sec)

八、集合类型

集合类型 SET 和枚举类似,也是得提前知道有多少个元素。SET 有以下特点:

1. 最大占用 8 Byte,int64。

2. 内部以二进制位的方式存储,对应的下标如果以十进制来看,就分别为 1,2,4,8,…,pow(2,63)。

3. 最大支持 64 个不同的元素,重复元素的插入,取出来直接去重。

4. 元素之间可以组合插入,比如下标为 1 和 2 的可以一起插入,直接插入 3 即可。

示例 9

定义表 c7 字段 c1 为 set 类型,包含了 8 个值,也就是下表最大为 pow(2,7)。

  1. mysql-(ytt/3305)->create table c7(c1 set('mysql','oracle','dble','postgresql','mongodb','redis','db2','sql server'));

  2. Query OK, 0 rows affected (0.02 sec)

插入 1 到 128 的所有组合。

  1. mysql-(ytt/3305)->INSERT INTO c7

  2. WITH RECURSIVE ytt_number (cnt) AS (

  3. SELECT 1 AS cnt

  4. UNION ALL

  5. SELECT cnt + 1

  6. FROM ytt_number

  7. WHERE cnt < pow(2, 7)

  8. )

  9. SELECT *

  10. FROM ytt_number;

  11. Query OK, 128 rows affected (0.01 sec)

  12. Records: 128 Duplicates: 0 Warnings: 0

查询的时候也是直接用下标或者元素来查。

  1. mysql-(ytt/3305)->select * from c7 where c1 = 1;

  2. +-------+

  3. | c1 |

  4. +-------+

  5. | mysql |

  6. +-------+

  7. 1 row in set (0.00 sec)


  8. mysql-(ytt/3305)->select * from c7 where c1 = 'mysql';

  9. +-------+

  10. | c1 |

  11. +-------+

  12. | mysql |

  13. +-------+

  14. 1 row in set (0.00 sec)

九、数据类型在存储函数中的用法

函数里除了显式声明的变量外,默认 session 变量的数据类型很弱,随着给定值的不同随意转换。

示例 10

定义一个函数,返回两个给定参数的乘积。定义里有两个变量,一个是 v_tmp 显式定义为 int64,另外一个 @vresult 随着给定值的类型随意变换类型。

  1. DELIMITER $$

  2. CREATE DEFINER=`ytt`@`127.0.0.1` FUNCTION `ytt_sample_data_type`(`f1` INT, `f2` INT) RETURNS varchar(100) CHARSET latin1

  3. NO SQL

  4. begin

  5. declare v_tmp bigint default 0;

  6. set v_tmp = f1 * f2;

  7. set @v_result = concat('The result is: ''',v_tmp,'''.');

  8. return @v_result;

  9. end$$

  10. DELIMITER ;

简单调用下。

  1. mysql-(ytt/3305)->select ytt_sample_data_type(1111,222) 'result';

  2. +--------------------------+

  3. | result |

  4. +--------------------------+

  5. | The result is: '246642'. |

  6. +--------------------------+

  7. 1 row in set (0.00 sec)

总结

本篇把 MySQL 基本的数据类型做了简单的介绍,并且用了一些容易理解的示例来梳理这些类型。我们在实际场景中,建议选择适合最合适的类型,不建议所有数据类型简单的最大化原则。比如能用 varchar(100),不用 varchar(1000)。

关于 MySQL 的技术内容,你们还有什么想知道的吗?赶紧留言告诉小编吧!

mysql更改字段类型_mysql修改字段数据类型_mysql中修改字段类型

想看更多技术好文,点个“在看”吧!

原创文章,作者:筱凯,如若转载,请注明出处:https://www.jingyueyun.com/ask/201.html

(0)
筱凯筱凯
上一篇 2024 年 7 月 10 日
下一篇 2024 年 7 月 10 日

相关推荐

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注

云产品限时秒杀。精选云产品高防服务器,500M大带宽限量抢购  >>点击进入