数据库(四):数据类型

进击のpython

*****

数据库——数据类型


数据库的数据类型主要是针对这几点进行讲解:

数值类型,日期类型,字符串类型,枚举与集合类型

学完之后相信你就能够对int和char有个更清晰的了解!


数值类型

整数类型

整型:tinyint smallint mediumint int bigint

小数:float:位数短不准

​ double:位数长不准

​ decimal:贼准

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

其实这些类型仅仅是代表着数字的范围不同

类型 大小 范围(有符号) 范围(无符号)
tinyint 1字节 (-128,127) (0,255)
smallint 2字节 (-32768,32767) (0,65535)
mediumint 3字节 (-8388608,8388607) (0,16777215)
int 4字节 (-2147483648,2147483647) (0,4294967295)
bigint 8字节 (-9223372036854775808,9223372036854775807) (0,18446744073709551615)
float 4字节 (-2147483648,2147483647) (0,4294967295)
double 8字节 (-9223372036854775808,9223372036854775807) (0,18446744073709551615)
decimal

int的范围是2**(4*8-1),因为符号占一个位,所以减一,那当没有符号的时候就是2**(4*8)-1

要是存储的数据超过限定值呢???

mysql> create table t1(id tinyint);
Query OK, 0 rows affected (0.93 sec)

mysql> show tables;
+----------------+
| Tables_in_text |
+----------------+
| t1             |
+----------------+
1 row in set (0.08 sec)

mysql> desc t1;
+-------+------------+------+-----+---------+-------+
| Field | Type       | Null | Key | Default | Extra |
+-------+------------+------+-----+---------+-------+
| id    | tinyint(4) | YES  |     | NULL    |       |
+-------+------------+------+-----+---------+-------+
1 row in set (0.51 sec)

mysql> insert into t1 values(-129),(128);
Query OK, 2 rows affected, 2 warnings (0.58 sec)
Records: 2  Duplicates: 0  Warnings: 2

mysql> select * from t1;
+------+
| id   |
+------+
| -128 |
|  127 |
+------+
2 rows in set (0.01 sec)

可以看到,当存储的值超过了限定值,那么就会存储边缘值入库

也可以设定没有符号:create table t2(id tinyint unsigned);

mysql> create table t2(id tinyint unsigned);
Query OK, 0 rows affected (1.01 sec)

mysql> desc t2;
+-------+---------------------+------+-----+---------+-------+
| Field | Type                | Null | Key | Default | Extra |
+-------+---------------------+------+-----+---------+-------+
| id    | tinyint(3) unsigned | YES  |     | NULL    |       |
+-------+---------------------+------+-----+---------+-------+
1 row in set (0.01 sec)

mysql> insert into t2 values(-129),(1000);
Query OK, 2 rows affected, 2 warnings (0.43 sec)
Records: 2  Duplicates: 0  Warnings: 2

mysql> desc t2;
+-------+---------------------+------+-----+---------+-------+
| Field | Type                | Null | Key | Default | Extra |
+-------+---------------------+------+-----+---------+-------+
| id    | tinyint(3) unsigned | YES  |     | NULL    |       |
+-------+---------------------+------+-----+---------+-------+
1 row in set (0.01 sec)

mysql> select * from t2;
+------+
| id   |
+------+
|    0 |
|  255 |
+------+
2 rows in set (0.00 sec)

结果也如我们预想一样~

那么坑就来了! char(数字)这么写过,那int(数字也是可以的)

int(数字)是可存储最大的数位对吧

我们来试一下create table t3(id int(1)); 应该是存一位数:

mysql> create table t3(id int(1));
Query OK, 0 rows affected (1.18 sec)

mysql> desc t3;
+-------+--------+------+-----+---------+-------+
| Field | Type   | Null | Key | Default | Extra |
+-------+--------+------+-----+---------+-------+
| id    | int(1) | YES  |     | NULL    |       |
+-------+--------+------+-----+---------+-------+
1 row in set (0.01 sec)

确实也像我们想的这样的表结构

现在就往里面填信息:

mysql> insert into t3 values(10000);
Query OK, 1 row affected (0.38 sec)

mysql> select * from t3;
+-------+
| id    |
+-------+
| 10000 |
+-------+
1 row in set (0.00 sec)

存没存进去????

原来啊,这个数值类型比较特殊,后面设置的数字,叫做显示宽度,存储位数根本就无法修改!

这个特性是数值类型特有的!!!!!!!别的不是这样的!!!!!

为了让你更深的理解,再说一个 zerofill 意思是用0填充

mysql> create table t4(id int(5) zerofill);
Query OK, 0 rows affected (0.94 sec)

mysql> insert into t4 values(1);
Query OK, 1 row affected (0.39 sec)

mysql> select * from t4;
+-------+
| id    |
+-------+
| 00001 |
+-------+
1 row in set (0.00 sec)

为什么是显示宽度呢?在这里显示宽度为5,不够宽度,前面用0占位

mysql> insert into t4 values(999999999999999999999999);
Query OK, 1 row affected, 2 warnings (0.38 sec)

mysql> select * from t4;
+------------+
| id         |
+------------+
|      00001 |
| 4294967295 |
+------------+
2 rows in set (0.00 sec)

当数据超过五位,那该存什么数据还是什么数据~~

当然你也应该留意过,不写的时候有个默认的长度10

因为最大位数就是10位!机智吧~~所以以后就不用指定了

小数没什么说的,这里就提一下就行:小数都有两个参数,第一个是最大数字总数,第二个是最大小数位数

float:总数255位 小数30位,前面不准确

double:总数255位 小数30位,后面不准确,更精确

decimal:总数65位,小数30位,一直准确,本质存的是字符串

日期类型

create table student(id int,name char(5),born_year year,birth_date date,class_time time,reg_time datetime);

mysql> create table student(id int,name char(5),born_year year,birth_date date,class_time time,reg_time datetime);
Query OK, 0 rows affected (0.83 sec)

mysql> desc student;
+------------+----------+------+-----+---------+-------+
| Field      | Type     | Null | Key | Default | Extra |
+------------+----------+------+-----+---------+-------+
| id         | int(11)  | YES  |     | NULL    |       |
| name       | char(5)  | YES  |     | NULL    |       |
| born_year  | year(4)  | YES  |     | NULL    |       |
| birth_date | date     | YES  |     | NULL    |       |
| class_time | time     | YES  |     | NULL    |       |
| reg_time   | datetime | YES  |     | NULL    |       |
+------------+----------+------+-----+---------+-------+
6 rows in set (0.06 sec)

这样就创建好了一个student表

接下来填数据

insert into student values (1,'ponny',now(),now(),now(),now());

mysql> select * from student;
+------+-------+-----------+------------+------------+---------------------+
| id   | name  | born_year | birth_date | class_time | reg_time            |
+------+-------+-----------+------------+------------+---------------------+
|    1 | ponny |      2019 | 2019-08-31 | 09:53:32   | 2019-08-31 09:53:32 |
+------+-------+-----------+------------+------------+---------------------+
1 row in set (0.00 sec)

now()是MySQL提供的获取当前时分秒年月日的方法,此时表的数据类型是year,代表着只接受年这个数据

当然你也可以自己插入,一个一个传值的那种,就不写了

当然除了datetime,还有一种时间类型叫做timestamp,那这两个方法到底有什么区别呢?

第一点就在于支持的时间不同,datetime的日期范围是1001——9999年,timestamp支持1970——2038年

第二点就是datetime存储数据占八个字节,timestamp占四个字节,所以datetime虽然占空间,但是考虑到时间的范围

还是选择datetime更好一点

那既然知道区别了,下面就研究一下各个数据类型的特点吧:

YEAR:YYYY(1901/2155)

DATE:YYYY-MM-DD(1001-01-01/9999-12-31)

TIME:HH:MM:SS('838:59:59'/'838:59:59')

DATETIME:YYYY-MM-DD HH:MM:SS(1001-01-01 00:00:00/9999-12-31 23:59:59)

TIMESTAMP:YYYYMMDD HHMMSS(1970-01-01 00:00:00/2037 某时)

字符类型

这个类型只需要了解两个就行,一个是char,一个是varchar

char指的是定长的字符,varchr指的是变长的字符

create table t5(name char(5));

create table t6(name varchar(5));

(这个括号里的数字可是实打实的规定存储字符的长度)

在这就要开始解释什么是定长,什么是变长了

定长就决定了,如果传的字符不够五个,就用空格给我补上

变长就决定了,如果传的字符不够五个,传的什么放进去什么

我们可以验证一下:

insert into t5 values("he");

insert into t6 values("he");

直接select吗?我觉得不是很行(你可以试一下),因为不够的字符都是空格补充的

你想知道有多少个空格是属于数据库的是很难的

所以我们才采用一种迂回的办法,查看返回值的长度

select char_length(name) from t5;

select char_length(name) from t6;

mysql> select char_length(name) from t5;
+-------------------+
| char_length(name) |
+-------------------+
|                 2 |
+-------------------+
1 row in set (0.00 sec)

mysql>
mysql> select char_length(name) from t6;
+-------------------+
| char_length(name) |
+-------------------+
|                 2 |
+-------------------+
1 row in set (0.00 sec)

但是这个MySQL很骚,骚就骚在,数据存储的时候有空格,当你查看的时候,就把空格去掉了

那为了不让去掉应该有的空格,我们需要设置一下SQL

SET sql_mode = 'PAD_CHAR_TO_FULL_LENGTH';

这会再执行查看语句就会发现他“原形毕露”!

mysql> select char_length(name) from t5;
+-------------------+
| char_length(name) |
+-------------------+
|                 5 |
+-------------------+
1 row in set (0.00 sec)

mysql>
mysql> select char_length(name) from t6;
+-------------------+
| char_length(name) |
+-------------------+
|                 2 |
+-------------------+
1 row in set (0.00 sec)

再多说一句,存储数据的时候会自动的把末尾的空格去掉,看清楚是末尾的!!

而在模糊查询的时候就不会去掉了,这个我们后面说

那定长和边长变长到底有什么区别呢?

如果我想存一组数据 ponny18女学生

利用char存起来就应该是:

ponny|18 |女 |学生 |

当我想拿的时候就是五个五个一取,没有问题

但是对于varchar来说,就没办法,因为变长,导致存的数据不知道多长,就可能会取到18女这种数据

所以,变长在存取数据的同时,在前面还专门的用一个字符来存储这个数据的长度

5+ponny|2+18|1+女|2+学生

官网上的图解是这样的:

所以说,单纯的说char比varchar浪费空间是不对的,如果正好在限制值上,char反而会更省空间

而且,一个字节最多存储255个数,也就意味着,字符的长度超过255个,varchar要再产生一个字节来存储个数

同时字符类型的存储是有长度上限的,所以像存储那些大的数据,一般存储的是个链接,然后指向另一个服务器

一个专门存放文件之类大文件的服务器

总结一下:char存储更快,但是占地有点大,由于现在更追求速度,所以最好使用char!

枚举类型&集合类型

枚举类型就是一堆里选一个当做数据

create table t7(id int,name char(6),sex enum('male','female'));

mysql> create table t7(id int,name char(6),sex enum('male','female'));
Query OK, 0 rows affected (1.00 sec)

mysql> desc t7;
+-------+-----------------------+------+-----+---------+-------+
| Field | Type                  | Null | Key | Default | Extra |
+-------+-----------------------+------+-----+---------+-------+
| id    | int(11)               | YES  |     | NULL    |       |
| name  | char(6)               | YES  |     | NULL    |       |
| sex   | enum('male','female') | YES  |     | NULL    |       |
+-------+-----------------------+------+-----+---------+-------+
3 rows in set (0.04 sec)

对表进行赋值:

mysql> insert into t7 values(1,'ponny','male');
Query OK, 1 row affected (0.39 sec)

mysql> select * from t7;
+------+-------+------+
| id   | name  | sex  |
+------+-------+------+
|    1 | ponny | male |
+------+-------+------+
1 row in set (0.00 sec)

那要是传的不是规定的数据呢?

mysql> insert into t7 values(2,'tony','other');
Query OK, 1 row affected, 1 warning (0.08 sec)

mysql> select * from t7;
+------+-------+------+
| id   | name  | sex  |
+------+-------+------+
|    1 | ponny | male |
|    2 | tony  |      |
+------+-------+------+
2 rows in set (0.00 sec)

可以看到,如果传的值不规范,就不会将该数据记录在表中

而集合类型就是多选多

mysql> alter table t7 add (hobby set("sing","dance","Rap"));
Query OK, 0 rows affected (1.34 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> desc t7;
+-------+---------------------------+------+-----+---------+-------+
| Field | Type                      | Null | Key | Default | Extra |
+-------+---------------------------+------+-----+---------+-------+
| id    | int(11)                   | YES  |     | NULL    |       |
| name  | char(6)                   | YES  |     | NULL    |       |
| sex   | enum('male','female')     | YES  |     | NULL    |       |
| hobby | set('sing','dance','Rap') | YES  |     | NULL    |       |
+-------+---------------------------+------+-----+---------+-------+
4 rows in set (0.05 sec)

mysql> insert into t7 values(3,"Tom","male","dance,Rap");
Query OK, 1 row affected (0.37 sec)

mysql> select * from t7;
+------+-------+------+-----------+
| id   | name  | sex  | hobby     |
+------+-------+------+-----------+
|    1 | ponny | male | NULL      |
|    2 | tony  |      | NULL      |
|    3 | Tom   | male | dance,Rap |
+------+-------+------+-----------+
3 rows in set (0.00 sec)

*****
*****
posted @ 2019-08-27 00:00  吃夏天的西瓜  阅读(1750)  评论(2编辑  收藏  举报