MYSQL中如何选择合适的数据类型MySQL 数据类型细分下来,大概有以下几类:
数值 , 典型代表为 tinyint,int,bigint
浮点/定点,典型代表为 float,double,decimal 以及相关的同义词
字符串,典型代表为 char,varchar
时间日期,典型代表为 date,datetime,time,timestamp
二进制,典型代表为 binary,varbinary
位类型
枚举类型
集合类型
大对象,比如 text,blob
json 文档类型
一、数值类型(不是数据类型,别看错了)如果用来存放整数,根据范围的不同,选择不同的类型 。
以上是几个整数选型的例子 。整数的应用范围最广泛 , 可以用来存储数字,也可以用来存储时间戳,还可以用来存储其他类型转换为数字后的编码,如 IPv4 等 。示例 1用 int32 来存放 IPv4 地址 , 比单纯用字符串节省空间 。表 x1,字段 ipaddr,利用函数 inet_aton , 检索的话用函数 inet_ntoa 。
查看磁盘空间占用,t3 占用最大 , t1 占用最小 。所以说如果整数存储范围有固定上限,并且未来也没有必要扩容的话 , 建议选择最小的类型,当然了对其他类型也适用 。root@ytt-pc:/var/lib/mysql/3305/ytt# ls -sihl总用量 3.0G3541825 861M -rw-r----- 1 mysql mysql 860M 12月 10 11:36 t1.ibd3541820 989M -rw-r----- 1 mysql mysql 988M 12月 10 11:38 t2.ibd3541823 1.2G -rw-r----- 1 mysql mysql 1.2G 12月 10 11:39 t3.ibd
二、浮点数 / 定点数先说 浮点数,float 和 double 都代表浮点数,区别简单记就是 float 默认占 4 Byte 。float(p) 中的 p 代表整数位最小精度 。如果 p24 则直接转换为 double,占 8 Byte 。p 最大值为 53,但最大值存在计算不精确的问题 。再说 定点数,包括 decimal 以及同义词 numeric,定点数的整数位和小数位分别存储 , 有效精度最大不能超过 65 。所以区别于 float 的在于精确存储,必须需要精确存储或者精确计算的最好定义为 decimal 即可 。示例 3创建一张表 y1,分别给字段 f1,f2,f3 不同的类型 。mysql-(ytt/3305)-create table y1(f1 float,f2 double,f3 decimal(10,2));Query OK, 0 rows affected (0.03 sec)
三、字符类型字符类型和整形一样 , 用途也很广 。用来存储字符、字符串、MySQL 所有未知的类型 。可以简单说是万能类型!
char(10) 代表最大支持 10 个字符存储,varhar(10) 虽然和 char(10) 可存储的字符数一样多,不同的是 varchar 类型存储的是实际大?。?char 存储的理论固定大小 。具体的字节数和字符集相关 。示例 4例如下面表 t4 , 两个字段 c1,c2,分别为 char 和 varchar 。mysql-(ytt/3305)-create table t4 (c1 char(20),c2 varchar(20));Query OK, 0 rows affected (0.02 sec)
所以在 char 和 varchar 选型上 , 要注意看是否合适的取值范围 。比如固定长度的值,肯定要选择 char;不确定的值,则选择 varchar 。
四、日期类型日期类型包含了 date,time,datetime,timestamp,以及 year 。year 占 1 Byte , date 占 3 Byte 。
time,timestamp,datetime 在不包含小数位时分别占用 3 Byte,4 Byte,8 Byte;小数位部分另外计算磁盘占用,见下面表格 。
请点击输入图片描述
请点击输入图片描述
请点击输入图片描述
注意:timestamp 代表的时间戳是一个 int32 存储的整数,取值范围为 '1970-01-01 00:00:01.000000' 到 '2038-01-19 03:14:07.999999';datetime 取值范围为 '1000-01-01 00:00:00.000000' 到 '9999-12-31 23:59:59.999999' 。
综上所述,日期这块类型的选择遵循以下原则:
1. 如果时间有可能超过时间戳范围 , 优先选择 datetime 。2. 如果需要单独获取年份值,比如按照年来分区 , 按照年来检索等,最好在表中添加一个 year 类型来参与 。3. 如果需要单独获取日期或者时间,最好是单独存放,而不是简单的用 datetime 或者 timestamp 。后面检索时,再加函数过滤,以免后期增加 SQL 编写带来额外消耗 。
4. 如果有保存毫秒类似的需求,最好是用时间类型自己的特性,不要直接用字符类型来代替 。MySQL 内部的类型转换对资源额外的消耗也是需要考虑的 。
示例 5
建立表 t5 , 对这些可能需要的字段全部分离开,这样以后写 SQL 语句的时候就很容易了 。
当然了 , 这种情形占用额外的磁盘空间 。如果想在易用性与空间占用量大这两点来折中 , 可以用 MySQL 的虚拟列来实时计算 。比如假设 c5 字段不存在,想要得到 c5 的结果 。mysql-(ytt/3305)-alter table t5 drop c5, add c5 year generated always as (year(c1)) virtual;Query OK, 1 row affected (2.46 sec)Records: 1Duplicates: 0Warnings: 0
五、二进制类型
binary 和 varbinary 对应了 char 和 varchar 的二进制存储 , 相关的特性都一样 。不同的有以下几点:
binary(10)/varbinary(10) 代表的不是字符个数,而是字节数 。
行结束符不一样 。char 的行结束符是 \0 , binary 的行结束符是 0x00 。
由于是二进制存储,所以字符编码以及排序规则这类就直接无效了 。
示例 6
来看这个 binary 存取的简单示例,还是之前的变量 @a 。
切记!这里要提前计算好 @a 占用的字节数,以防存储溢出 。
六、位类型
bit 为 MySQL 里存储比特位的类型,最大支持 64 比特位, 直接以二进制方式存储,一般用来存储状态类的信息 。比如 , 性别,真假等 。具有以下特性:
1. 对于 bit(8) 如果单纯存放 1 位 , 左边以 0 填充 00000001 。2. 查询时可以直接十进制来过滤数据 。3. 如果此字段加上索引 , MySQL 不会自己做类型转换,只能用二进制来过滤 。
示例 7
创建表 c1, 字段性别定义一个比特位 。mysql-(ytt/3305)-create table c1(gender bit(1));Query OK, 0 rows affected (0.02 sec)
mysql-(ytt/3305)-select cast(gender as unsigned)'f1' from c1; ------ | f1| ------ |0 ||1 | ------ 2 rows in set (0.00 sec)
过滤数据也一样,二进制或者直接十进制都行 。mysql-(ytt/3305)-select conv(gender,16,10) as gender \- from c1 where gender = b'1';-------- | gender | -------- | 1| -------- 1 row in set (0.00 sec)mysql-(ytt/3305)-select conv(gender,16,10) as gender \- from c1 where gender = '1'; -------- | gender | -------- | 1| -------- 1 row in set (0.00 sec)
其实这样的场景 , 也可以定义为 char(0),这也是类似于 bit 非常优化的一种用法 。
mysql-(ytt/3305)-create table c2(gender char(0));Query OK, 0 rows affected (0.03 sec)
那现在我给表 c1 简单的造点测试数据 。
mysql-(ytt/3305)-select count(*) from c1; ---------- | count(*) | ---------- | 33554432 | ---------- 1 row in set (1.37 sec)
把 c1 的数据全部插入 c2 。
mysql-(ytt/3305)-insert into c2 select if(gender = 0,'',null) from c1;Query OK, 33554432 rows affected (2 min 18.80 sec)Records: 33554432Duplicates: 0Warnings: 0
两张表的磁盘占用差不多 。root@ytt-pc:/var/lib/mysql/3305/ytt# ls -sihl总用量 1.9G4085684 933M -rw-r----- 1 mysql mysql 932M 12月 11 10:16 c1.ibd4082686 917M -rw-r----- 1 mysql mysql 916M 12月 11 10:22 c2.ibd
检索方式稍微有些不同,不过效率也差不多 。所以说 , 字符类型不愧为万能类型 。
七、枚举类型
枚举类型,也即 enum 。适合提前规划好了所有已经知道的值,且未来最好不要加新值的情形 。枚举类型有以下特性:
1. 最大占用 2 Byte 。2. 最大支持 65535 个不同元素 。3. MySQL 后台存储以下标的方式,也就是 tinyint 或者 smallint 的方式,下标从 1 开始 。4. 排序时按照下标排序,而不是按照里面元素的数据类型 。所以这点要格外注意 。
示例 8
创建表 t7 。mysql-(ytt/3305)-create table t7(c1 enum('mysql','oracle','dble','postgresql','mongodb','redis','db2','sql server'));Query OK, 0 rows affected (0.03 sec)
八、集合类型
集合类型 SET 和枚举类似,也是得提前知道有多少个元素 。SET 有以下特点:
1. 最大占用 8 Byte,int64 。2. 内部以二进制位的方式存储,对应的下标如果以十进制来看,就分别为 1,2,4,8,...,pow(2,63) 。3. 最大支持 64 个不同的元素,重复元素的插入,取出来直接去重 。4. 元素之间可以组合插入,比如下标为 1 和 2 的可以一起插入,直接插入 3 即可 。
示例 9
【mysql怎么定义类型 mysql定义数组】定义表 c7 字段 c1 为 set 类型,包含了 8 个值,也就是下表最大为 pow(2,7) 。
mysql-(ytt/3305)-create table c7(c1 set('mysql','oracle','dble','postgresql','mongodb','redis','db2','sql server'));Query OK, 0 rows affected (0.02 sec)
插入 1 到 128 的所有组合 。
mysql-(ytt/3305)-INSERT INTO c7WITH RECURSIVE ytt_number (cnt) AS (SELECT 1 AS cntUNION ALLSELECT cnt1FROM ytt_numberWHERE cntpow(2, 7))SELECT *FROM ytt_number;Query OK, 128 rows affected (0.01 sec)Records: 128Duplicates: 0Warnings: 0
九、数据类型在存储函数中的用法
函数里除了显式声明的变量外,默认 session 变量的数据类型很弱,随着给定值的不同随意转换 。
示例 10
定义一个函数,返回两个给定参数的乘积 。定义里有两个变量 , 一个是 v_tmp 显式定义为 int64,另外一个 @vresult 随着给定值的类型随意变换类型 。
简单调用下 。
mysql-(ytt/3305)-select ytt_sample_data_type(1111,222) 'result'; -------------------------- | result| -------------------------- | The result is: '246642'. | -------------------------- 1 row in set (0.00 sec)
总结
本篇把 MySQL 基本的数据类型做了简单的介绍,并且用了一些容易理解的示例来梳理这些类型 。我们在实际场景中,建议选择适合最合适的类型,不建议所有数据类型简单的最大化原则 。比如能用 varchar(100),不用 varchar(1000) 。
mysql 如何定义date 类型 例子以每24小时作为一份时间(而非自然日)mysql怎么定义类型,根据用户mysql怎么定义类型的配置有两种工作模式:带状模式中,用户仅定义开始日期时,从开始日期(含)开始,每份时间1个分片地无限增加下去;环状模式中,用户定义了开始日期和结束日期时,以结束日期(含)和开始日期(含)之间的时间份数作为分片总数(分片数量固定) , 以类似取模的方式路由到这些分片里 。
1. DBLE 启动时,读取用户在 rule.xml 配置的 sBeginDate 来确定起始时间
2. 读取用户在 rule.xml 配置的 sPartionDay 来确定每个 MySQL 分片承载多少天内的数据
3. 读取用户在 rule.xml 配置的 dateFormat 来确定分片索引的日期格式
4. 在 DBLE 的运行过程中,用户访问使用这个算法的表时,WHERE 子句中的分片索引值(字符串),会被提取出来尝试转换成 Java 内部的时间类型
5. 然后求分片索引值与起始时间的差,除以 MySQL 分片承载的天数,确定所属分片
1. DBLE 启动时,读取用户在 rule.xml 配置的起始时间 sBeginDate、终止时间 sEndDate 和每个 MySQL 分片承载多少天数据 sPartionDay
2. 根据用户设置,建立起以 sBeginDate 开始 , 每 sPartionDay 天一个分片,直到 sEndDate 为止的一个环,把分片串联串联起来
3. 读取用户在 rule.xml 配置的 defaultNode
4. 在 DBLE 的运行过程中,用户访问使用这个算法的表时,WHERE 子句中的分片索引值(字符串),会被提取出来尝试转换成 Java 内部的日期类型
5. 然后求分片索引值与起始日期的差:如果分片索引值不早于 sBeginDate(哪怕晚于 sEndDate),就以 MySQL 分片承载的天数为模数,对分片索引值求模得到所属分片;如果分片索引值早于 sBeginDate,就会被放到 defaultNode 分片上
与MyCat的类似分片算法对比
中间件
DBLE
MyCat
分片算法种类date 分区算法按日期(天)分片
两种中间件的取模范围分片算法使用上无差别
开发注意点
【分片索引】1. 必须是字符串,而且 java.text.SimpleDateFormat 能基于用户指定的 dateFormat 来转换成 java.util.Date
【分片索引】2. 提供带状模式和环状模式两种模式
【分片索引】3. 带状模式以 sBeginDate(含)起,以 86400000 毫秒(24 小时整)为一份,每 sPartionDay 份为一个分片 , 理论上分片数量可以无限增长,但是出现 sBeginDate 之前的数据而且没有设定 defaultNode 的话,会路由失败(如果有 defaultNode,则路由至 defaultNode)
【分片索引】4. 环状模式以 86400000 毫秒(24 小时整)为一份 , 每 sPartionDay 份为一个分片,以 sBeginDate(含)到 sEndDate(含)的时间长度除以单个分片长度得到恒定的分片数量,但是出现 sBeginDate 之前的数据而且没有设定 defaultNode 的话 , 会路由失败(如果有 defaultNode , 则路由至 defaultNode)
【分片索引】5. 无论哪种模式,分片索引字段的格式化字符串 dateFormat 由用户指定
【分片索引】6. 无论哪种模式,划分不是以日历时间为准,无法对应自然月和自然年,且会受闰秒问题影响
运维注意点
【扩容】1. 带状模式中,随着 sBeginDate 之后的数据出现 , 分片数量的增加无需再平衡
【扩容】2. 带状模式没有自动增添分片的能力,需要运维手工提前增加分片;如果路由策略计算出的分片并不存在时,会导致失败
【扩容】3. 环状模式中 , 如果新旧 [sBeginDate,sEndDate] 之间有重叠 , 需要进行部分数据迁移;如果新旧 [sBeginDate,sEndDate] 之间没有重叠 , 需要数据再平衡
配置注意点
【配置项】1. 在 rule.xml 中 , 可配置项为 propertyname="sBeginDate" 、 propertyname="sPartionDay" 、 propertyname="dateFormat" 、 propertyname="sEndDate" 和 propertyname="defaultNode"
【配置项】2.在 rule.xml 中配置 propertyname="dateFormat" , 符合 java.text.SimpleDateFormat 规范的字符串,用于告知 DBLE 如何解析sBeginDate和sEndDate
【配置项】3.在 rule.xml 中配置 propertyname="sBeginDate",必须是符合 dateFormat 的日期字符串
【配置项】4.在 rule.xml 中配置 propertyname="sEndDate",必须是符合 dateFormat 的日期字符串;配置了该项使用的是环状模式,若没有配置该项则使用的是带状模式
【配置项】5.在 rule.xml 中配置 propertyname="sPartionDay",非负整数,该分片策略以 86400000 毫秒(24 小时整)作为一份,而 sPartionDay 告诉 DBLE 把每多少份放在同一个分片
【配置项】6.在 rule.xml 中配置 propertyname="defaultNode" 标签,非必须配置项,不配置该项的话,用户的分片索引值没落在 mapFile 定义
MySQL存储过程里怎么定义一个参数类型和表的变量类型一样?MySQL存储过程中 , 定义变量有两种方式:
? 1、使用set或select直接赋值,变量名以@开头,可以在一个会话(即连接)的任何地方声明,作用域是整个会话,称为用户变量 。例如:set @var=1;
? 2、 以declare关键字声明的变量,只能在存储过程中使用 , 称为存储过程变量,主要用在存储过程中,或者是给存储传参数中 。例如: declare var1 int default 0;
两者的区别是:
? ? 在调用存储过程时,以declare声明的变量都会被初始化为null 。而会话变量(即@开头的变量)则不会被再初始化,在一个会话(连接)内,只须初始化一次,之后在会话内都是对上一次计算的结果,就相当于在是这个会话内的全局变量 。
如何设置mysql中的数据类型使用ALTER命令修改,格式:
ALTER TABLE 表名MODIFY COLUMN 字段名 字段类型定义;
例如:
ALTER TABLE chatter_usersMODIFY COLUMN ip VARCHAR(50);
mysql类型MySQL常见数据类型:[数值]、[日期时间]和[字符串]类型 。
一、数值
1、整型
MySQL数据类型含义(有符号)tinyint1个字节
范围(-128~127)smallint2个字节
范围(-32768~32767)mediumint3个字节
范围(-8388608~8388607)int4个字节
范围(-2147483648~2147483647)bigint8个字节
范围( -9.22*10的18次方)
在int类型里,都是用来存储整形数据,可以根据实际需要选取数据类型 。
取值范围如果加了unsigned,则最大数值范围翻倍,比如 tinyint unsigned的取值范围为(0~256) 。
int(m)里的m是表示SELECT查询结果集中的显示宽度,无实际意义,不影响实际的取值范围
2、浮点型
MySQL数据类型含义float(m,d)单精度浮点型
8位精度(4字节)
m总个数,d小数位double(m,d)双精度浮点型
16位精度(8字节)
m总个数,d小数位decimal(m,d)定点数
总个数m38, d小数位
设一个字段定义为float(5,2) , 如果插入一个数123.45678,实际数据库里存的是123.46(四舍五入)
关于mysql怎么定义类型和mysql定义数组的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站 。
推荐阅读
- js数组内容是文字去重的简单介绍
- 在山区如何做推广工作,山区农产品如何推广
- 简单java代码_,java 简单代码
- php整站带数据 php网站统计
- 微信小程序php交互源码,微信小程序php接口
- java时间服务器,java获取服务器时间
- 鸿蒙os2.0智慧屏界面,鸿蒙智慧屏体验
- 怎样看到java源代码 如何查看java源代码
- linux系统命令改ip,linux命令行改ip