DML常用操作
DML常用操作
DML(Data Manipulation Language)数据操作语言,以INSERT、UPDATE、DELETE三种指令为核心,
分别代表插入、更新与删除,是必须要掌握的指令,DML和SQL中的select熟称CRUD(增删改查),均属针对表的操作
插入数据(insert)
单行插入
方法1:
insert into 表名[(字段,字段)] values (值,值);
说明:
值和字段需要一一对应
如果是字符型或者日期型,值需要用单引号引起来;如果是正事就不需要
字段和值的个数必须一致,且位置对应
字段如果不能为空,则必须插入值
为空的字段可以不用插入值,但要注意:字段和值都不写,或,字段写上,值用null代替
表名后的字段可以不写,不写标识所有字段,顺序与表中字段一致
方法2:
insert into 表名 set 字段 = 值,字段 = 值;
这种方式不建议使用,官方也不推荐
mysql> select * from test1;
+---+---+
| a | b |
+---+---+
| 1 | 0 |
+---+---+
1 row in set (0.00 sec)
mysql> insert into test1 set a=2,b=2;
Query OK, 1 row affected (0.00 sec)
mysql> select * from test1;
+---+---+
| a | b |
+---+---+
| 1 | 0 |
| 2 | 2 |
+---+---+
2 rows in set (0.00 sec)
批量插入
方法1:
insert into 表名 [(字段,字段)] values (值,值),(值,值),(值,值);
方法2:
insert into 表 [(字段,字段)]
数据来源select语句;
#例如
insert into test1 (a,b) select 字段a,字段b from test2;
说明:
数据来源select语句可以有多种写法,需要注意select返回的结果和插入数据的字段数量、顺序、类型需要一致。
例子:
mysql> drop table if exists test1;
Query OK, 0 rows affected (0.00 sec)
mysql> create table test1(a int,b int);
Query OK, 0 rows affected (0.00 sec)
mysql> drop table if exists test2;
Query OK, 0 rows affected (0.01 sec)
mysql> create table test2(c1 int,c2 int,c3 int);
Query OK, 0 rows affected (0.01 sec)
#test2中插入数据
mysql> insert into test2 values (100,101,102),(200,201,202),(300,301,302),(400,401,402);
Query OK, 4 rows affected (0.00 sec)
Records: 4 Duplicates: 0 Warnings: 0
#test1中插入数据
mysql> insert into test1 (a,b) values(1,1),(2,3),(3,3);
Query OK, 3 rows affected (0.01 sec)
Records: 3 Duplicates: 0 Warnings: 0
#test1中插入数据,数据来源test2
mysql> insert into test1 (a,b) select c2,c3 from test2 where c1>=200;
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 1 | 1 |
| 2 | 3 |
| 3 | 3 |
| 201 | 202 |
| 301 | 302 |
| 401 | 402 |
+------+------+
6 rows in set (0.00 sec)
更新数据
要求不复杂建议用单表更新,易于维护
单表更新
语法:
update 表名 [[as] 别名] set [别名.]字段 = 值,[别名.]字段 = 值 [where条件];
示例:
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 1 | 1 |
| 2 | 3 |
| 3 | 3 |
| 201 | 202 |
| 301 | 302 |
| 401 | 402 |
+------+------+
6 rows in set (0.00 sec)
mysql> update test1 as TEST1 set a=2;
Query OK, 5 rows affected (0.01 sec)
Rows matched: 6 Changed: 5 Warnings: 0
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 2 | 1 |
| 2 | 3 |
| 2 | 3 |
| 2 | 202 |
| 2 | 302 |
| 2 | 402 |
+------+------+
6 rows in set (0.00 sec)
#as可以省略
mysql> update test1 t set t.a=3;
Query OK, 6 rows affected (0.00 sec)
Rows matched: 6 Changed: 6 Warnings: 0
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 3 | 1 |
| 3 | 3 |
| 3 | 3 |
| 3 | 202 |
| 3 | 302 |
| 3 | 402 |
+------+------+
6 rows in set (0.00 sec)
mysql> update test1 t set a=4,b=5;
Query OK, 6 rows affected (0.00 sec)
Rows matched: 6 Changed: 6 Warnings: 0
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 4 | 5 |
| 4 | 5 |
| 4 | 5 |
| 4 | 5 |
| 4 | 5 |
| 4 | 5 |
+------+------+
6 rows in set (0.00 sec)
#需要根据条件来更新就使用where
多表更新
同时更新多个表
语法:
update 表1 [[as] 别名1],表名2 [[as] 别名2]
set [别名.]字段 = 值,[别名.]字段 = 值
[where条件]
示例:
#无别名
update test1,test2 set test1.a = 2 ,test1.b = 2, test2.c1 = 10;
update test1,test2 set test1.a = 2 ,test1.b = 2, test2.c1 = 10 where test1.a = test2.c1;
#使用别名更新,as可省略,但建议写上
update test1 t1,test2 t2 set t1.a = 2 ,t1.b = 2, t2.c1 = 10 where t1.a = t2.c1;
update test1 as t1,test2 t2 set t1.a = 2 ,t1.b = 2, t2.c1 = 10 where t1.a = t2.c1;
删除操作
delete删除
delete单表删除
delete [别名] from 表名 [[as] 别名] [where条件];
#可以加上更多的可选条件
delete from table_name [WHERE clause] [ORDER BY 'column_name' [DESC]] [LIMIT [m,]n];
别名可以选择省略不写
如果有别名,delete后面必须写别名
如果没有别名,delete后面的别名可以省略不写。
也就是说不用别名时,是可以直接delete from的
示例
-- 删除test1表所有记录
delete from test1;
-- 删除test1表所有记录
delete test1 from test1;
-- 有别名的方式,删除test1表所有记录
delete t1 from test1 t1;
-- 有别名的方式删除满足条件的记录
delete t1 from test1 t1 where t1.a>100;
多表删除
同时删除多个表中的记录
语法:
delete [别名1,别名2] from 表1 [[as] 别名1],表2 [[as] 别名2] [where条件];
#不写别名时
delete [表1,表2] from 表1,表2 [where条件];
别名可以省略不写,但是需要在delete后面跟上表名,多个表名之间用号隔开。
示例:
delete t1 from test1 t1,test2 t2 where t1.a=t2.c2;
删除test1中记录,条件是 test1的a字段和test2中c2字段相同
运行效果:
drop table if exists test1;
drop table if exists test2;
#建表
create table test1(a int,b int);
create table test2(c1 int,c2 int,c3 int);
#插入数据
#test2中插入数据
insert into test2 values (100,101,102),(200,201,202),(300,301,302),(400,401,402);
#test1中插入数据
insert into test1 (a,b) select 1,1 union all select 2,2 union all select 2,2;
#向test1插入数据,数据来源于test2表
insert into test1 (a,b) select c2,c3 from test2 where c1>=200;
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 1 | 1 |
| 2 | 2 |
| 2 | 2 |
| 201 | 202 |
| 301 | 302 |
| 401 | 402 |
+------+------+
6 rows in set (0.00 sec)
mysql> select * from test2;
+------+------+------+
| c1 | c2 | c3 |
+------+------+------+
| 100 | 101 | 102 |
| 200 | 201 | 202 |
| 300 | 301 | 302 |
| 400 | 401 | 402 |
+------+------+------+
4 rows in set (0.00 sec)
mysql> delete t1 from test1 t1,test2 t2 where t1.a=t2.c2;
Query OK, 3 rows affected (0.00 sec)
#不写别名
delete test1 from test1,test2 where test1.a=test2.c2;
mysql> select * from test1;
+------+------+
| a | b |
+------+------+
| 1 | 1 |
| 2 | 2 |
| 2 | 2 |
+------+------+
3 rows in set (0.00 sec)
mysql> select * from test2;
+------+------+------+
| c1 | c2 | c3 |
+------+------+------+
| 100 | 101 | 102 |
| 200 | 201 | 202 |
| 300 | 301 | 302 |
| 400 | 401 | 402 |
+------+------+------+
4 rows in set (0.00 sec)
#test1中的三条记录被删除了
同时删除多个表,也就是delete后面带上要删的表名或者别名
示例:
delete t1,t2 from test1 t1,test2 t2 where t1.a=t2.c2;
在实际使用中还是常用delete from 表名
这种用法,对于单表而言可以这样随意写,多表的话delete后面必须带要删除的表名或表别名,否则有语法错误
truncate删除
语法:
truncate 表名;
drop, truncate, delete的区别
delete是删除表中指定条件的数据;
truncate是删除表中所有数据;
drop是删除表数据和表结构
-
drop(删除表):删除类容和定义,释放空间,简单来说就是把整个表去掉,以后要新增数据是不可能的,除非新增一个表。
drop语句将删除表的结构被依赖的约束(constrain),触发器(trigger)索引(index),依赖于该表的存储过程/函数将被保留,但其状态会变为:invalid。
-
truncate (清空表中的数据):删除内容、释放空间但不删除定义(保留表的数据结构),与drop不同的是,只是清空表数据而已。
注意:truncate不能删除具体行数据,要删就要把整个表清空了。
-
delete (删除表中的数据):delete 语句用于删除表中的行。delete语句执行删除的过程是每次从表中删除一行,并且同时将该行的删除操作作为事务记录在日志中保存,以便进行进行回滚操作。
truncate与不带where的delete :只删除数据,而不删除表的结构(定义)
truncate table 删除表中的所有行,但表结构及其列、约束、索引等保持不变
对于由foreign key约束引用的表,不能使用truncate table ,而应使用不带where子句的delete语句。由于truncate table 记录在日志中,所以它不能激活触发器。
delete语句是数据库操作语言(dml),这个操作会放到 rollback segement 中,事务提交之后才生效;如果有相应的 trigger,执行的时候将被触发。
truncate、drop 是数据库定义语言(ddl),操作立即生效,原数据不放到 rollback segment 中,不能回滚,操作不触发 trigger。
如果有自增列,truncate方式删除之后,自增列的值会被初始化,delete方式要分情况(如果数据库被重启了,自增列值也会被初始化,数据库未被重启,则不变)
-
如果要删除表定义及其数据,请使用 drop table 语句
-
安全性:小心使用 drop 和 truncate,尤其没有备份的时候,否则后悔都来不及
-
删除速度,一般来说: drop> truncate > delete
drop | truncate | drop | |
---|---|---|---|
条件删除 | 不支持 | 不支持 | 支持 |
删除表结构 | 支持 | 不支持 | 不支持 |
事务的方式删除 | 不支持 | 不支持 | 支持 |
删除触发器 | 否 | 否 | 是 |
三种删除方式的原理
delete
DELETE from TABLE_NAME where xxx
1、DELETE属于数据库DML操作语言,只删除数据不删除表的结构,会走事务,执行时会触发trigger;
2、在 InnoDB 中,DELETE其实并不会真的把数据删除,mysql 实际上只是给删除的数据打了个标记为已删除,因此 delete 删除表中的数据时,表文件在磁盘上所占空间不会变小,存储空间不会被释放,只是把删除的数据行设置为不可见。 虽然未释放磁盘空间,但是下次插入数据的时候,仍然可以重用这部分空间(重用 → 覆盖)。
3、 DELETE执行时,会先将所删除数据缓存到rollback segement中,事务commit之后生效;
4、 delete from table_name删除表的全部数据,对于MyISAM 会立刻释放磁盘空间,InnoDB 不会释放磁盘空间;
5、对于delete from table_name where xxx 带条件的删除, 不管是InnoDB还是MyISAM都不会释放磁盘空间;
6、 delete操作以后使用 optimize table table_name 会立刻释放磁盘空间。不管是InnoDB还是MyISAM 。所以要想达到释放磁盘空间的目的,delete以后执行optimize table 操作。
7、delete 操作是一行一行执行删除的,并且同时将该行的的删除操作日志记录在redo和undo表空间中以便进行回滚(rollback)和重做操作,生成的大量日志也会占用磁盘空间。
truncate
Truncate table TABLE_NAME
1、truncate:属于数据库DDL定义语言,不走事务,原数据不放到 rollback segment 中,操作不触发 trigger。
执行后立即生效,无法找回 执行后立即生效,无法找回 执行后立即生效,无法找回
2、 truncate table table_name 立刻释放磁盘空间 ,不管是 InnoDB和MyISAM 。truncate table其实有点类似于drop table 然后creat,只不过这个create table 的过程做了优化,比如表结构文件之前已经有了等等。所以速度上应该是接近drop table的速度;
3、truncate能够快速清空一个表。并且重置auto_increment的值。
但对于不同的类型存储引擎需要注意的地方是:
- 对于MyISAM,truncate会重置auto_increment(自增序列)的值为1。而delete后表仍然保持auto_increment。
- 对于InnoDB,truncate会重置auto_increment的值为1。delete后表仍然保持auto_increment。但是在做delete整个表之后重启MySQL的话,则重启后的auto_increment会被置为1。
也就是说,InnoDB的表本身是无法持久保存auto_increment。delete表之后auto_increment仍然保存在内存,但是重启后就丢失了,只能从1开始。实质上重启后的auto_increment会从 SELECT 1+MAX(ai_col) FROM t 开始。
4、 小心使用 truncate,包括drop,尤其没有备份的时候,如果误删除线上的表,flush logs
,跟领导反应,当然也可以删库跑路
drop
Drop table Tablename
1、drop:属于数据库DDL定义语言,同Truncate;
执行后立即生效,无法找回 执行后立即生效,无法找回 执行后立即生效,无法找回
2、 drop table table_name 立刻释放磁盘空间 ,不管是 InnoDB 和 MyISAM; drop 语句将删除表的结构被依赖的约束(constrain)、触发器(trigger)、索引(index); 依赖于该表的存储过程/函数将保留,但是变为 invalid 状态。