当前位置:网站首页>【MySQL基础】-【数据处理之增删改】
【MySQL基础】-【数据处理之增删改】
2022-08-05 10:27:00 【CaraYQ】
目录
数据处理之增删改
在atguigudb数据库中新建emp1表:
USE atguigudb;
CREATE TABLE IF NOT EXISTS emp1(
id INT,
`name` VARCHAR(15),
hire_date DATE,
salary DOUBLE(10,2)
);
DESC emp1;
SELECT *
FROM emp1;
添加数据
一、方式一:一条一条的添加数据
- 没有指明添加的字段
#正确的
INSERT INTO emp1
VALUES (1,'Tom','2000-12-21',3400); #注意:一定要按照声明的字段的先后顺序添加
#错误的
INSERT INTO emp1
VALUES (2,3400,'2000-12-21','Jerry');
- 指明要添加的字段 (推荐)
INSERT INTO emp1(id,hire_date,salary,`name`)
VALUES(2,'1999-09-09',4000,'Jerry');
# 说明:没有进行赋值的hire_date 的值为 null
INSERT INTO emp1(id,salary,`name`)
VALUES(3,4500,'shk');
- 同时插入多条记录 (推荐)
INSERT INTO emp1(id,NAME,salary)
VALUES
(4,'Jim',5000),
(5,'张俊杰',5500);
二、方式二:将查询结果插入到表中
SELECT * FROM emp1;# 查看一下emp1的数据
# 将查询结果插入到emp1表中
INSERT INTO emp1(id,NAME,salary,hire_date)
SELECT employee_id,last_name,salary,hire_date # 查询的字段一定要与添加到的表的字段一一对应
FROM employees
WHERE department_id IN (70,60);
通过查看emp1、employees的结构我们发现数据长度发生了变化
DESC emp1;
DESC employees;
说明:emp1表中要添加数据的字段的长度不能低于employees表中查询的字段的长度。如果emp1表中要添加数据的字段的长度低于employees表中查询的字段的长度的话,就有添加不成功的风险。
更新数据 (或修改数据)
一、语法:UPDATE .... SET .... WHERE ...
UPDATE emp1
SET hire_date = CURDATE()
WHERE id = 5;
二、同时修改一条数据的多个字段
UPDATE emp1
SET hire_date = CURDATE(),salary = 6000
WHERE id = 4;
三、如果没有写约束条件WHERE ...
,那就是批量修改数据
UPDATE emp1
SET hire_date = CURDATE();# 将表中hire_date字段的值全改为当前时间
四、题目:将表中姓名中包含字符a的提薪20%
UPDATE emp1
SET salary = salary * 1.2
WHERE NAME LIKE '%a%';
五、修改数据时,是可能存在不成功的情况的。(可能是由于约束的影响造成的)
UPDATE employees
SET department_id = 10000
WHERE employee_id = 102;
删除数据
一、语法:DELETE FROM .... WHERE....
DELETE FROM emp1
WHERE id = 1;
二、在删除数据时,也有可能因为约束的影响,导致删除失败
DELETE FROM departments
WHERE department_id = 50;
三、小结:DML操作默认情况下,执行完以后都会自动提交数据。如果希望执行完以后不自动提交数据,则需要使用SET autocommit = FALSE;
MySQL8的新特性:计算列
某一列的值是通过别的列计算得来的。例如,a列值为1、b列值为2,c列不需要手动插入,定义a+b的结果为c的值,那么c就是计算列,是通过别的列计算得来的。
USE atguigudb;
CREATE TABLE test1(
a INT,
b INT,
c INT GENERATED ALWAYS AS (a + b) VIRTUAL #在声明时就指定字段c是计算列
);
INSERT INTO test1(a,b)
VALUES(10,20);
SELECT * FROM test1;
#更改a的值,发现c的值也自动变了
UPDATE test1
SET a = 100;
DDL和DML的综合案例
# 1、创建数据库test01_library
CREATE DATABASE IF NOT EXISTS test01_library;
USE test01_library;
# 2、创建表 books,表结构如下:
CREATE TABLE IF NOT EXISTS books(
id INT,
`name` VARCHAR(50),
`authors` VARCHAR(100),
price FLOAT,
pubdate YEAR,
note VARCHAR(100),
num INT
);
# 3、向books表中插入记录
# 1)不指定字段名称,插入第一条记录
INSERT INTO books VALUES (1, "Tal of AAA", "Dickes", 23, 1995, "novel", 11);
# 2)指定所有字段名称,插入第二记录
INSERT INTO books(id, `name`, `authors`, price, pubdate, note, num) VALUES (2, "EmmaT", "Jane lura", 35, 1993, "joke", 22);
# 3)同时插入多条记录(剩下的所有记录)
INSERT INTO books
VALUES
(3, "Story of Jane", "Jane Tim", 40, 2001, "novel", 0),
(4, "Lovey Day", "George Byron", 20, 2005, "novel", 30),
(5, "Old land", "Honore Blade", 30, 2010, "law", 0),
(6, "The Battle", "Upton Sara", 30, 1999, "medicine", 40),
(7, "Rose Hood", "Richard haggard", 28, 2008, "cartoon", 28);
SELECT * FROM books;
# 4、将小说类型(novel)的书的价格都增加5。
UPDATE books SET price = price + 5 WHERE note = "novel";
# 5、将名称为EmmaT的书的价格改为40,并将说明改为drama。
UPDATE books SET price = 40, note = "drama" WHERE `name`= "EmmaT";
# 6、删除库存为0的记录。
DELETE FROM books WHERE num = 0;
# 7、统计书名中包含a字母的书
SELECT COUNT(*)
FROM books
WHERE `name` LIKE "%a%";
# 8、统计书名中包含a字母的书的数量和库存总量
SELECT num, SUM(num) "sum"
FROM books
WHERE `name` LIKE "%a%";
# 9、找出“novel”类型的书,按照价格降序排列
SELECT * FROM books WHERE note = "novel" ORDER BY price DESC;
# 10、查询图书信息,按照库存量降序排列,如果库存量相同的按照note升序排列
SELECT * FROM books ORDER BY num DESC, note;
# 11、按照note分类统计书的数量
SELECT note, COUNT(*) FROM books GROUP BY note;
# 12、按照note分类统计书的库存量,显示库存量超过30本的
SELECT note, SUM(num) FROM books GROUP BY note HAVING SUM(num) > 30;
# 13、查询所有图书,每页显示5本,显示第二页
SELECT * FROM books LIMIT 5,5;
# 14、按照note分类统计书的库存量,显示库存量最多的
SELECT note, SUM(num) FROM books GROUP BY note ORDER BY SUM(num) DESC LIMIT 0,1;
# 15、查询书名达到10个字符的书,不包括里面的空格
SELECT `name` FROM books WHERE CHAR_LENGTH(REPLACE(`name`," ","")) >= 10;
# 16、查询书名和类型,其中note值为novel显示小说,law显示法律,medicine显示医药,
#cartoon显示卡通,joke显示笑话
SELECT `name`, note, CASE note WHEN "novel" THEN "小说"
WHEN "law" THEN "法律"
WHEN "medicine" THEN "医药"
WHEN "cartoon" THEN "卡通"
WHEN "joke" THEN "笑话"
ELSE "其他" END "类型"
FROM books;
# 17、查询书名、库存,其中num值超过30本的,显示滞销,大于0并低于10的,
#显示畅销,为0的显示需要无货
SELECT `name`, num, CASE WHEN num > 30 THEN "滞销"
WHEN num = 0 THEN "无货"
WHEN num > 0 AND num < 10 THEN "畅销"
ELSE "正常" END "note"
FROM books;
# 18、统计每一种note的库存量,并合计总量
# IFNULL(note,'合计库存总量'):如果note为null,则命名为合计库存总量,否则使用note的值作为行名
SELECT IFNULL(note,'合计库存总量') AS note,SUM(num)
FROM books
GROUP BY note WITH ROLLUP;# WITH ROLLUP将你计算出来的数量再进行一次累加
# 19、统计每一种note的数量,并合计总量
SELECT IFNULL(note,'合计总量') AS note,COUNT(*)
FROM books
GROUP BY note WITH ROLLUP;
# 20、统计库存量前三名的图书
SELECT * FROM books ORDER BY num DESC LIMIT 0,3;
# 21、找出最早出版的一本书
SELECT * FROM books ORDER BY pubdate LIMIT 0,1;
# 22、找出novel中价格最高的一本书
SELECT * FROM books WHERE `note` = "novel" ORDER BY price DESC LIMIT 0 ,1;
# 23、找出书名中字数最多的一本书,不含空格
SELECT * FROM books ORDER BY LENGTH(REPLACE(`name`," ","")) DESC LIMIT 0,1;
MySQL数据类型精讲
一、MySQL中的数据类型
二、常见数据类型的属性
字符集设置
一、语法:character set name
二、创建数据库时指名字符集:CREATE DATABASE IF NOT EXISTS dbtest12 CHARACTER SET 'utf8';
三、创建表的时候,指名表的字符集:CREATE TABLE temp(id INT) CHARACTER SET 'utf8';
四、创建表,指名表中的字段时,可以指定字段的字符集:CREATE TABLE temp1(id INT,NAME VARCHAR(15) CHARACTER SET 'gbk');
五、查看:
SHOW VARIABLES LIKE 'character_%';# 查看字符集
SHOW CREATE DATABASE dbtest12;
SHOW CREATE TABLE temp;
SHOW CREATE TABLE temp1;
六、如果字段没有声明字符集,就使用字段所在表的字符集,如果表没有声明字符集,就使用表所在数据库的字符集,如果数据库没有指定字符集,就是用my.ini配置文件中配置的字符集
整数类型
一、整数类型一共有5种:TINYINT、SMALLINT、MEDIUMINT、INT(INTEGER)和BIGINT,他们的区别如下:
USE dbtest12;
CREATE TABLE test_int1(
f1 TINYINT,
f2 SMALLINT,
f3 MEDIUMINT,
f4 INTEGER,
f5 BIGINT
);
DESC test_int1;
INSERT INTO test_int1(f1)
VALUES(12),(-12),(-128),(127);
SELECT * FROM test_int1;
#Out of range value for column 'f1' at row 1
INSERT INTO test_int1(f1)
VALUES(128);
在MySQL8.0和MySQL5.7中执行DESC test_int1;
展示的结果不一样:
括号中的数字代表显示数据的宽度,如TINYINT数据类型的有符号数取值范围为-128到127,-128一共4位,-是符号位也要算在里面,127应该是+127,也是4位
二、可选属性
- M:表示显示宽度,M的取值范围是(0, 255)。例如,
int(5)
:当数据宽度小于5位的时候在数字前面需要用字符填满宽度。该项功能需要配合ZEROFILL
使用,表示用0
填满宽度,否则指定显示宽度无效。
CREATE TABLE test_int2(
f1 INT,
f2 INT(5),
f3 INT(5) ZEROFILL
# 显示宽度为5。当insert的值不足5位时,使用0填充。
# 如果f3插入的值超过5位,则f1、f2、f3是一样的,即有没有INT(5) ZEROFILL都一样
)
INSERT INTO test_int2(f1,f2)
VALUES(123,123),(123456,123456);
INSERT INTO test_int2(f3)
VALUES(123),(123456);
SELECT * FROM test_int2;
执行SHOW CREATE TABLE test_int2;
后发现:当使用ZEROFILL
时,自动会添加UNSIGNED
,即f3的值只能是正数
注意:显示宽度不会对插入的数据有任何影响,还是按照类型的实际宽度进行保存,即显示宽度与类型可以存储的值范围无关 。从MySQL 8.0.17开始,整数数据类型不推荐使用显示宽度属性。
- UNSIGNED: 无符号类型(非负),所有的整数类型都有一个可选的属性UNSIGNED(无符号属性),无符号整数类型的最小取值为0。所以,如果需要在MySQL数据库中保存非负整数值时,可以将整数类型设置为无符号类型。
CREATE TABLE test_int3(
f1 INT UNSIGNED
);
DESC test_int3;
INSERT INTO test_int3
VALUES(2412321);
#Out of range value for column 'f1' at row 1
INSERT INTO test_int3
VALUES(4294967296);
int类型默认显示宽度为int(11)
,无符号int类型默认显示宽度为int(10)
。
- ZEROFILL:: 0填充,(如果某列是ZEROFILL,那么MySQL会自动为当前列添加UNSIGNED属性),如果指定了ZEROFILL只是表示不够M位时,用0在左边填充,如果超过M位,只要不超过数据存储范围即可。
三、适用场景
TINYINT :一般用于枚举数据,比如系统设定取值范围很小且固定的场景。
SMALLINT :可以用于较小范围的统计数据,比如统计工厂的固定资产库存数量等。
MEDIUMINT :用于较大整数的计算,比如车站每日的客流量等。
INT、INTEGER :取值范围足够大,一般情况下不用考虑超限问题,用得最多。比如商品编号。
BIGINT :只有当你处理特别巨大的整数时才会用到。比如双十一的交易量、大型门户网站点击量、证券公司衍生产品持仓等。
四、如何选择:在评估用哪种整数类型的时候,你需要考虑存储空间和可靠性的平衡问题:一方面,用占用字节数少的整数类型可以节省存储空间;另一方面,要是为了节省存储空间, 使用的整数类型取值范围太小,一旦遇到超出取值范围的情况,就可能引起 系统错误 ,影响可靠性。在实际工作中,系统故障产生的成本远远超过增加几个字段存储空间所产生的成本。因此,我建议你首先确保数据不会超过取值范围,在这个前提之下,再去考虑如何节省存储空间。
浮点类型
一、MySQL支持的浮点数类型:FLOAT、DOUBLE、REAL。
- FLOAT 表示单精度浮点数;
- DOUBLE 表示双精度浮点数;
- REAL默认就是DOUBLE。如果你把SQL模式设定为启用
REAL_AS_FLOAT
,那么,MySQL就认为REAL是FLOAT。如果要启用REAL_AS_FLOAT
,可以通过以下SQL语句实现:SET sql_mode = “REAL_AS_FLOAT”;
二、FLOAT和DOUBLE的区别:FLOAT占用字节数少,取值范围小;DOUBLE占用字节数多,取值范围也大。
三、为什么浮点数类型的无符号数取值范围,只相当于有符号数取值范围的一半,也就是只相当于有符号数取值范围大于等于零的部分呢?因为MySQL 存储浮点数的格式为: 符号(S) 、 尾数(M)和阶码(E) 。因此,无论有没有符号,MySQL的浮点数都会存储表示符号的部分。因此, 所谓的无符号数取值范围,其实就是有符号数取值范围大于等于零的部分。
CREATE TABLE test_double1(
f1 FLOAT,
f2 FLOAT(5,2),
f3 DOUBLE,
f4 DOUBLE(5,2)
);
DESC test_double1;
INSERT INTO test_double1(f1,f2)
VALUES(123.45,123.45);
SELECT * FROM test_double1;
INSERT INTO test_double1(f3,f4)
VALUES(123.45,123.456);
#f4 DOUBLE(5,2)规定小数位2位,这里插入的数据是3位,会存在四舍五入,最终存入的数据位123.46
#Out of range value for column 'f4' at row 1
INSERT INTO test_double1(f3,f4)
VALUES(123.45,1234.456);
#f4 DOUBLE(5,2)规定整数位3位,这里插入的数据是4位,直接报错
#Out of range value for column 'f4' at row 1
INSERT INTO test_double1(f3,f4)
VALUES(123.45,999.995);
#f4 DOUBLE(5,2)规定整数位3位,小数位2位,这里插入的数据小数位是3位,四舍五入后进位导致整数位变成4位,超出规定,直接报错
四、FLOAT和DOUBLE的精度问题
CREATE TABLE test_double2(
f1 DOUBLE
);
INSERT INTO test_double2
VALUES(0.47),(0.44),(0.19);
SELECT SUM(f1)# 按理说应该等于1.1,但实际上等于1.0999999999。
# 你也可以尝试把数据类型改成FLOAT,然后运行求和查询,得到的是1.0999999940395355。显然,误差更大了。
FROM test_double2;
SELECT SUM(f1) = 1.1,1.1 = 1.1# false true
FROM test_double2;
浮点数类型有个缺陷,就是不精准。出现上述问题的原因:采用二进制的方式来进行存储数据。比如9.625,用二进制来表达就是1001.101,或者表达成1.001101×2^3。如果尾数不是0或5(比如9.624),你就无法用一个二进制数来精确表达。进而,就只好在取值允许的范围内进行四舍五入。
在编程中,如果用到浮点数,要特别注意误差问题,因为浮点数是不准确的,所以我们要避免使用=
来判断两个数是否相等。同时,在一些对精确度要求较高的项目中,千万不要使用浮点数,不然会导致结果错误,甚至是造成不可挽回的损失。那么,MySQL有没有精准的数据类型呢?当然有,这就是定点数类型: DECIMAL 。
定点数类型
一、定点数类型只有DECIMAL一种类型
二、DECIMAL(M,D)的最大取值范围与DOUBLE类型一样,但是有效的数据范围是由M和D决定的。DECIMAL 的存储空间并不是固定的,由精度值M决定,总共占用的存储空间为M+2个字节。也就是说,在一些对精度要求不高的场景下,比起占用同样字节长度的定点数,浮点数表达的数值范围可以更大一些。
三、定点数在MySQL内部是以字符串的形式进行存储,这就决定了它一定是精准的。
四、当DECIMAL类型不指定精度和标度时,其默认为DECIMAL(10,0)。当数据的精度超出了定点数类型的精度范围时,则MySQL同样会进行四舍五入处理。
五、定点数与浮点数的区别:
- 浮点数相对于定点数的优点是在长度一定的情况下,浮点类型取值范围大,但是不精准,适用于需要取值范围大,又可以容忍微小误差的科学计算场景(比如计算化学、分子建模、流体动力学等)
- 定点数类型取值范围相对小,但是精准,没有误差,适合于对精度要求极高的场景 (比如涉及金额计算的场景)
CREATE TABLE test_decimal1(
f1 DECIMAL,
f2 DECIMAL(5,2)
);
DESC test_decimal1;
INSERT INTO test_decimal1(f1)
VALUES(123),(123.45);
SELECT * FROM test_decimal1;
INSERT INTO test_decimal1(f2)
VALUES(999.99);
INSERT INTO test_decimal1(f2)
VALUES(67.567);#存在四舍五入
#Out of range value for column 'f2' at row 1
INSERT INTO test_decimal1(f2)
VALUES(1267.567);
#Out of range value for column 'f2' at row 1
INSERT INTO test_decimal1(f2)
VALUES(999.995);
六、由于 DECIMAL 数据类型的精准性,在项目中,除了极少数(比如商品编号)用到整数类型外,其他的数值都用的是DECIMAL,原因就是这个项目所处的零售行业,要求精准,一分钱也不能差
ALTER TABLE test_double2
MODIFY f1 DECIMAL(5,2);
DESC test_double2;
SELECT SUM(f1)
FROM test_double2;
SELECT SUM(f1) = 1.1,1.1 = 1.1# true true
FROM test_double2;
位类型:BIT
一、BIT类型中存储的是二进制值,类似010110。
BIT类型,如果没有指定(M),默认是1位。这个1位,表示只能存1位的二进制值。这里(M)是表示二进制的位数,位数最小值为1,最大值为64。
CREATE TABLE test_bit1(
f1 BIT,
f2 BIT(5),# 最大存储数值位31
f3 BIT(64)
);
DESC test_bit1;
INSERT INTO test_bit1(f1)
VALUES(0),(1);
SELECT *
FROM test_bit1;
#Data too long for column 'f1' at row 1
INSERT INTO test_bit1(f1)
VALUES(2);
INSERT INTO test_bit1(f2)
VALUES(31);
#Data too long for column 'f2' at row 1
INSERT INTO test_bit1(f2)
VALUES(32);
二、使用SELECT命令查询位字段时,可以用BIN()
或HEX()
函数进行读取。
# BIN()展示二进制存储的数据,HEX()展示十六进制存储的数据
SELECT BIN(f1),BIN(f2),HEX(f1),HEX(f2)
FROM test_bit1;
#此时+0以后,可以以十进制的方式显示数据
SELECT f1 + 0, f2 + 0
FROM test_bit1;
日期与时间类型
一、MySQL8.0版本支持的日期和时间类型主要有:
- YEAR类型通常用来表示年
- DATE类型通常用来表示年、月、日
- TIME类型通常用来表示时、分、秒
- DATETIME类型通常用来表示年、月、日、时、分、秒
- TIMESTAMP类型通常用来表示带时区的年、月、日、时、分、秒
二、类型 名称 字节 日期格式 最小值 最大值
三、为什么时间类型 TIME 的取值范围不是-23:59:59~23:59:59呢?原因是MySQL设计的TIME类型,不光表示一天之内的时间,而且可以用来表示一个时间间隔,这个时间间隔可以超过24小时。
YEAR类型
一、在MySQL中,YEAR有以下几种存储格式:
- 以4位字符串或数字格式表示YEAR类型,其格式为YYYY,最小值为1901,最大值为2155。
- 以2位字符串格式表示YEAR类型,最小值为00,最大值为99。
(1)当取值为01到69时,表示2001到2069;
(2)当取值为70到99时,表示1970到1999;
(3)当取值整数的0或00添加的话,那么是0000年;
(4)当取值是日期/字符串的’0’添加的话,是2000年。
二、从MySQL5.5.27开始,2位格式的YEAR已经不推荐使用。YEAR默认格式就是YYYY
,没必要写成YEAR(4)
CREATE TABLE test_year(
f1 YEAR,
f2 YEAR(4)
);
DESC test_year;
INSERT INTO test_year(f1)
VALUES('2021'),(2022);
SELECT * FROM test_year;
INSERT INTO test_year(f1)
VALUES ('2155');
#Out of range value for column 'f1' at row 1
INSERT INTO test_year(f1)
VALUES ('2156');
INSERT INTO test_year(f1)
VALUES ('69'),('70');
INSERT INTO test_year(f1)
VALUES (0),('00');
DATE类型
一、DATE类型表示日期,没有时间部分,格式为YYYY-MM-DD
,其中,YYYY
表示年份,MM
表示月份,DD
表示日期。需要3个字节的存储空间。在向DATE类型的字段插入数据时,同样需要满足一定的格式条件。
二、以YYYY-MM-DD
格式或者YYYYMMDD
格式表示的字符串日期,其最小取值为1000-01-01
,最大取值为9999-12-03
。YYYYMMDD
格式会被转化为YYYY-MM-DD
格式。
三、以YY-MM-DD
格式或者YYMMDD
格式表示的字符串日期,此格式中,年份为两位数值或字符串满足YEAR类型的格式条件为:当年份取值为00到69时,会被转化为2000到2069;当年份取值为70到99时,会被转化为1970到1999。
CREATE TABLE test_date1(
f1 DATE
);
DESC test_date1;
INSERT INTO test_date1
VALUES ('2020-10-01'), ('20201001'),(20201001);
INSERT INTO test_date1
VALUES ('00-01-01'), ('000101'), ('69-10-01'), ('691001'), ('70-01-01'), ('700101'), ('99-01-01'), ('990101');
INSERT INTO test_date1
VALUES (000301), (690301), (700301), (990301); #存在隐式转换
四、使用CURRENT_DATE()
或者NOW()
函数,会插入当前系统的日期。
INSERT INTO test_date1
VALUES (CURDATE()),(CURRENT_DATE()),(NOW());
SELECT *
FROM test_date1;
TIME类型
一、TIME类型用来表示时间,不包含日期部分。在MySQL中,需要3个字节的存储空间来存储TIME类型的数据,可以使用HH:MM:SS
格式来表示TIME类型,其中,HH表示小时,MM表示分钟,SS表示秒。
二、在MySQL中,向TIME类型的字段插入数据时,也可以使用几种不同的格式。
- 可以使用带有冒号的字符串,比如
D HH:MM:SS
、HH:MM:SS
、HH:MM
、D HH:MM
、D HH
或SS
格式,都能被正确地插入TIME类型的字段中。其中D表示天,其最小值为0,最大值为34。如果使用带有D格式的字符串插入TIME类型的字段时,D会被转化为小时,计算格式为D*24+HH
。当使用带有冒号并且不带D的字符串表示时间时,表示当天的时间,比如12:10
表示12:10:00
,而不是00:12:10
。 - 可以使用不带有冒号的字符串或者数字,格式为
HHMMSS
或者HHMMSS
。如果插入一个不合法的字符串或者数字,MySQL在存储数据时,会将其自动转化为00:00:00
进行存储。比如1210
,MySQL会将最右边的两位解析成秒,表示00:12:10
,而不是12:10:00
。 - 使用
CURRENT_TIME()
或者NOW()
,会插入当前系统的时间。
CREATE TABLE test_time1(
f1 TIME
);
DESC test_time1;
INSERT INTO test_time1
VALUES('2 12:30:29'), ('12:35:29'), ('12:40'), ('2 12:40'),('1 05'), ('45');
INSERT INTO test_time1
VALUES ('123520'), (124011),(1210);
INSERT INTO test_time1
VALUES (NOW()), (CURRENT_TIME()),(CURTIME());
SELECT *
FROM test_time1;
DATETIME类型
一、在格式上为DATE类型和TIME类型的组合,可以表示为YYYY-MM-DD HH:MM:SS
,其中YYYY表示年份,MM表示月份,DD表示日期,HH表示小时,MM表示分钟,SS表示秒。
二、在向DATETIME类型的字段插入数据时,同样需要满足一定的格式条件。
- 以
YYYY-MM-DD HH:MM:SS
格式或者YYYYMMDDHHMMSS
格式的字符串插入DATETIME类型的字段时,最小值为1000-01-01 00:00:00
,最大值为9999-12-03 23:59:59
。 - 以
YYYYMMDDHHMMSS
格式的数字插入DATETIME类型的字段时,会被转化为YYYY-MM-DD HH:MM:SS
格式。 - 以
YY-MM-DD HH:MM:SS
格式或者YYMMDDHHMMSS
格式的字符串插
三、DATETIME类型的字段时,两位数的年份规则符合YEAR类型的规则,00到69表示2000到2069;70到99表示1970到1999。
四、使用函数CURRENT_TIMESTAMP()
和NOW()
,可以向DATETIME类型的字段插入系统的当前日期和时间。
CREATE TABLE test_datetime1(
dt DATETIME
);
INSERT INTO test_datetime1
VALUES ('2021-01-01 06:50:30'), ('20210101065030');
INSERT INTO test_datetime1
VALUES ('99-01-01 00:00:00'), ('990101000000'), ('20-01-01 00:00:00'), ('200101000000');
INSERT INTO test_datetime1
VALUES (20200101000000), (200101000000), (19990101000000), (990101000000);
INSERT INTO test_datetime1
VALUES (CURRENT_TIMESTAMP()), (NOW()),(SYSDATE());
SELECT *
FROM test_datetime1;
TIMESTAMP类型
一、TIMESTAMP类型也可以表示日期时间,其显示格式与DATETIME类型相同,都是YYYY-MM-DD HH:MM:SS
,需要4个字节的存储空间。但是TIMESTAMP存储的时间范围比DATETIME要小很多,只能存储“1970-01-01 00:00:01 UTC”到“2038-01-19 03:14:07 UTC”之间的时间。其中,UTC表示世界统一时间,也叫作世界标准时间。
二、存储数据的时候需要对当前时间所在的时区进行转换,查询数据的时候再将时间转换回当前的时区。因此,使用TIMESTAMP存储的同一个时间值,在不同的时区查询时会显示不同的时间。
三、向TIMESTAMP类型的字段插入数据时,当插入的数据格式满足YY-MM-DD HH:MM:SS
和YYMMDDHHMMSS
时,两位数值的年份同样符合YEAR类型的规则条件,只不过表示的时间范围要小很多。
四、如果向TIMESTAMP类型的字段插入的时间超出了TIMESTAMP类型的范围,则MySQL会抛出错误信息。
CREATE TABLE test_timestamp1(
ts TIMESTAMP
);
INSERT INTO test_timestamp1
VALUES ('1999-01-01 03:04:50'), ('19990101030405'), ('99-01-01 03:04:05'), ('990101030405');
INSERT INTO test_timestamp1
VALUES ('[email protected]@[email protected]@[email protected]'), ('[email protected]@[email protected]@[email protected]');
INSERT INTO test_timestamp1
VALUES (CURRENT_TIMESTAMP()), (NOW());
#Incorrect datetime value
INSERT INTO test_timestamp1
VALUES ('2038-01-20 03:14:07');
SELECT *
FROM test_timestamp1;
四、TIMESTAMP和DATETIME的区别:
- TIMESTAMP存储空间比较小,表示的日期时间范围也比较小
- 底层存储方式不同,TIMESTAMP底层存储的是毫秒值,距离
1970-1-1 0:0:0 0
毫秒的毫秒值。 - 两个日期比较大小或日期计算时,TIMESTAMP更方便、更快。
- TIMESTAMP和时区有关。TIMESTAMP会根据用户的时区不同,显示不同的结果。而DATETIME则只能反映出插入时当地的时区,其他时区的人查看数据必然会有误差的。
CREATE TABLE temp_time(
d1 DATETIME,
d2 TIMESTAMP
);
INSERT INTO temp_time VALUES('2021-9-2 14:45:52','2021-9-2 14:45:52');
INSERT INTO temp_time VALUES(NOW(),NOW());
SELECT * FROM temp_time;
#修改当前的时区
SET time_zone = '+9:00';
SELECT * FROM temp_time;
修改时区后二者显示不一样,这是因为TIMESTAMP会将时间转换成毫秒值存储,然后在展示时间时,会根据时区将毫秒值转换成时间,但是DATETIME不受时区的影响
五、开发中经验:用得最多的日期时间类型时DATETIME 。因为这个数据类型包括了完整的日期和时间信息,取值范围也最大,使用起来比较方便。毕竟,如果日期时间信息分散在好几个字段,很不容易记,而且查询的时候,SQL语句也会更加复杂。此外,一般存注册时间、商品发布时间等,不建议使用DATETIME存储,而是使用时间戳 ,因为DATETIME虽然直观,但不便于计算。
文本字符串类型
边栏推荐
- DFINITY 基金会创始人谈熊市沉浮,DeFi 项目该何去何从
- 公众号如何运维?公众号运维专业团队
- RT-Thread记录(一、RT-Thread 版本、RT-Thread Studio开发环境 及 配合CubeMX开发快速上手)
- 华为轻量级神经网络架构GhostNet再升级,GPU上大显身手的G-GhostNet(IJCV22)
- NowCoderTOP35-40 - continuous update ing
- The century-old Nordic luxury home appliance brand ASKO smart wine cabinet in the three-temperature area presents the Chinese Valentine’s Day, and tastes the love of the delicacy
- 基于MindSpore高效完成图像分割,实现Dice!
- 牛刀小试基本语法,Go lang1.18入门精炼教程,由白丁入鸿儒,go lang基本语法和变量的使用EP02
- How does the official account operate and maintain?Public account operation and maintenance professional team
- The query that the user's test score is greater than the average score of a single subject
猜你喜欢
告白数字化转型时代:麦聪软件以最简单的方式让企业把数据用起来
一文道清什么是SPL
PCB layout must know: teach you to correctly lay out the circuit board of the op amp
DFINITY 基金会创始人谈熊市沉浮,DeFi 项目该何去何从
012年通过修补_sss_提高扩散模型效率
华为轻量级神经网络架构GhostNet再升级,GPU上大显身手的G-GhostNet(IJCV22)
MySQL transactions
three objects are arranged in a spherical shape around the circumference
Ali's new launch: Microservices Assault Manual, all operations are written out in PDF
JS逆向入门学习之回收商网,手机号码简易加密解析
随机推荐
SMB + SMB2: Accessing shares return an error after prolonged idle period
NowCoderTOP35-40——持续更新ing
Offensive World-PWN-new_easypwn
什么是 DevOps?看这一篇就够了!
牛刀小试基本语法,Go lang1.18入门精炼教程,由白丁入鸿儒,go lang基本语法和变量的使用EP02
60行从零开始自己动手写FutureTask是什么体验?
MySQL data view
Complete image segmentation efficiently based on MindSpore and realize Dice!
Use KUSTO query statement (KQL) to query LOG on Azure Data Explorer Database
Is digital transformation a business buy-in?
如何修改管理工具client_encoding
机器学习-基础知识 - Precision, Recall, Sensitivity, Specificity, Accuracy, FNR, FPR, TPR, TNR, F1 Score, Bal
PHP operation mangoDb
一文道清什么是SPL
The century-old Nordic luxury home appliance brand ASKO smart wine cabinet in the three-temperature area presents the Chinese Valentine’s Day, and tastes the love of the delicacy
2022 Huashu Cup Mathematical Modeling Question A Optimization Design Ideas for Ring Oscillators Code Sharing
公众号如何运维?公众号运维专业团队
Jenkins manual (2) - software configuration
2022 Huashu Cup Mathematical Modeling Ideas Analysis and Exchange
单片机:温度控制DS18B20