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,执行的时候将被触发。

    truncatedrop 是数据库定义语言(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 状态。

posted @ 2021-02-03 14:21  EverEternity  阅读(2773)  评论(0编辑  收藏  举报