表的增删查改简称CRUD:Create(新增),Retrieve(查找),Update(修改),Delete(删除)。
CRUD的操作对象是对表当中的数据,是典型的DML语句(Data Manipulation Language 数据操作语言)。
语法:
INSERT [INTO] table_name [(column [, column] ...)] VALUES (value_list) [, (value_list)] ... // 解释: value_list: value, [, value] ...
说明:
为了方便进行演示,下面创建一个学生表,表当中包含自增长的主键id、学号、姓名和QQ号。如下:
create table students( id int unsigned primary key auto_increment, sn int unsigned not null unique comment '学号', name varchar(20) not null comment '姓名', qq varchar(15) unique comment 'qq号码' ); desc students;
单行数据 + 全列插入
当我们在使用insert语句向学生表中插入记录,每次向表中插入一条记录,并且插入记录时不指定column列表,表示按照表中默认的列顺序进行全列插入,因此插入的每条记录中的列值需要按表列顺序依次列出。如下:
insert into students values(100, 1000, '唐玄奘', null); insert into students values(101, 1023, '孙悟空', null); select * from students;
多行数据 + 指定列插入
使用insert语句也可以一次向表中插入多条记录,插入的多条记录之间使用逗号隔开,并且插入记录时可以只指定某些列进行插入。如下:
insert into students(sn, name) values (123, '曹操'), (187,'刘备'); select * from students;
插入否则更新
有时候我们在向表中插入新记录时,由于 主键 或者 唯一键 对应的值已经存在而导致我们插入失败。
这时如果我们还想插入我们的数据就可以选择性的进行同步更新操作:
插入否则更新的SQL语法如下:
INSERT ... ON DUPLICATE KEY UPDATE column1=value1 [, column2=value2] ...;
作用:
说明:
例如:这次我们继续向学生表中刚才的插入记录。如下:
insert into students (id,sn,name) values(100, 1000, '唐僧') on duplicate key update name= '唐僧', sn= 1010;
可以看到在冲突的情况下我们也确实更新了数据。
说明:执行插入否则更新的SQL后,可以通过受影响的数据行数来判断本次数据的插入情况:
替换数据
语法:
replace into students(sn, name) values(1023, '孙行者');
执行替换数据的SQL后,也可以通过受影响的数据行数来判断本次数据的插入情况:
查找其实是我们使用mysql最多的操作,其语法如下:
SELECT [DISTINCT] {* | {column1 [, column2] ...}} FROM table_name [WHERE ...] [ORDER BY ...] [LIMIT ...];
说明:
为了方便我们进行演示,下面创建一个成绩表,表当中包含:自增长的主键id、姓名、以及该同学的语文成绩、数学成绩和英语成绩。如下:
CREATE TABLE exam_result ( id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT, name VARCHAR(20) NOT NULL COMMENT '同学姓名', chinese float DEFAULT 0.0 COMMENT '语文成绩', math float DEFAULT 0.0 COMMENT '数学成绩', english float DEFAULT 0.0 COMMENT '英语成绩' );
接下来向表中插入几条测试记录,以供我们进行查找。如下:
INSERT INTO exam_result (name, chinese, math, english) VALUES ('唐三藏', 67, 98, 56), ('孙悟空', 87, 78, 77), ('猪悟能', 88, 98, 90), ('曹孟德', 82, 84, 67), ('刘玄德', 55, 85, 45), ('孙权', 70, 73, 78), ('宋公明', 75, 65, 30);
1、全列查询
通过全列查询我们能够拿到所有列的数据,在查询数据时直接用*代替column列表,表示进行全列查询。如下:
select * from exam_result;
通常情况下不建议使用* 进行全列查询
2、指定列查询
在查询数据时也可以只对指定的列进行查询,这时将需要查询的列在column列表列出即可。
例如我们现在要看一下班级中的数学成绩,如下:
select name, math from exam_result;
3、结果去重
当然,我们也可以在select后面跟上distinct,表示对结果去重,下面我们尝试对上面的7个数学成绩进行去重。
select distinct name, math from exam_result;
我们发现好像没有达到我们想象的效果,还是有两个相同的数据,这时为什么呢?
因为distinct是对结果进行去重,而我们结果中不仅有数学成绩还有姓名,这两个组合起来没有重复所以也就不会被去重了。
因此我们在对数学成绩进行去重时不应该加上姓名,使用下面的SQL进行去重:
select distinct math from exam_result;
可以看到我们的结果变成了6行(以前是7行),而且没有重复数据
4、查询字段为表达式
select语句是一个非常特殊的语句,其在查询数据时,column列表中除了能罗列表中存在的列名外,我们也可以将表达式罗列到column列表中。如下:
select name, math, 1+1 from exam_result;
因此select不仅能够用来查询数据,还可以用来计算某些表达式或执行某些函数。如下:
如果我们将表达式罗列到column列表中:
那么每当一条记录被筛选出来时就会执行这个表达式,然后将表达式的计算结果作为这条记录的一个列值进行显示。
column列表中的表达式可以包含表中已有的字段,这时每当一条记录被筛选出来时,就会将记录中对应的列值提供给表达式进行计算。
例如:我们将每个同学的数学成绩都加10分。
select name, math, math+10 from exam_result;
又或者我们可以求一下同学的总分:
select name, math+chinese+english from exam_result;
5、为查询结果指定别名
为查询结果指定别名的SQL语法如下:
SELECT column [AS] alias_name [...] FROM table_name;
说明:
比如刚才查询成绩表中的数据时,求总分后的名字为math+chinese+english,有一点丑陋,我们使用别名进行解决。如下:
select name, math+chinese+english as total from exam_result;
通过刚才的学习我们对一列的数据能够进行很好的操作了,但是有些时候,某一列的数据特别多,我们想要快速的找到我们想要的行,于是就需要有where条件了!
where子句添加与否的区别:
where子句中可以指明一个或多个筛选条件,各个筛选条件之间用逻辑运算符AND或OR进行关联,下面给出了where子句中常用的比较运算符和逻辑运算符。
比较运算符:
运算符 | 说明 |
---|---|
>、>=、<、<= | 大于、大于等于、小于、小于等于 |
= | 等于。NULL不安全,例如NULL=NULL的结果是NULL而不是TRUE(1) |
<=> | 等于。NULL安全,例如NULL<=>NULL的结果就是TRUE(1) |
!=、<> | 不等于,NULL不安全 |
BETWEEN a0 AND a1 | 范围匹配。如果a0<=value<=a1,则返回TRUE(1) |
IN(option1, option2, …) | 如果是IN中的任意一个option,则返回TRUE(1) |
IS NULL | 如果是NULL,则返回TRUE(1) |
IS NOT NULL | 如果不是NULL,则返回TRUE(1) |
LIKE | 模糊匹配。%表示任意多个字符(包括0个),_表示任意一个字符 |
逻辑运算符:
运算符 | 说明 |
---|---|
AND | 多个条件同时为TRUE(1),则结果为TRUE(1),否则为FALSE(0) |
OR | 任意一个条件为TRUE(1),则结果为TRUE(1),否则为FALSE(0) |
NOT | 条件为TRUE(1),则结果为FALSE(0);条件为FALSE(0),则结果为TRUE(1) |
下面我们来做一些案例帮助我们理解并掌握这些运算符:
1、查询英语不及格的同学及其英语成绩
在where子句中指明筛选条件为英语成绩小于60,我们直接使用<运算符即可。如下:
select name, english from exam_result where english < 60;
2、查询语文成绩在80到90分的同学及其语文成绩
在where子句中指明筛选条件为语文成绩,然后根据分数直接>=80 ,<= 90即可,然后根据其并列关系条件我们选择and逻辑运算符进行连接 。如下:
select name, chinese from exam_result where chinese >= 80 and chinese <= 90;
此外,这里也可以使用BETWEEN a0 AND a1来指明语文成绩的的所在区间。如下:
select name, chinese from exam_result where chinese between 80 and 90;
3、数学成绩是 58 或者 59 或者 98 或者 99 分的同学及数学成绩
在where子句中指明筛选条件为数学成绩等于58或59或98或99,在select的column列表中指明要查询的列为姓名和数学成绩。如下:
select name ,math from exam_result where math=58 or math=59 or math=98 or math=99;
此外,这里也可以通过IN(58, 59, 98, 99)的方式来判断数学成绩是否符合筛选要求。如下:
select name, math from exam_reault where math in (58,59,98,99);
4、姓孙的同学 及 孙某同学
这里给出的查询条件是模糊条件,所以在where子句中通过模糊匹配来判断当前同学是否姓孙(需要用到%来匹配多个字符),在select的column列表中指明要查询的列为姓名。如下:
select name from exam_result where name like '孙%';
这里给出的查询条件是同样是模糊条件,在where子句中通过模糊匹配来判断当前同学是否为孙某(需要用到_来严格匹配单个字符),在select的column列表中指明要查询的列为姓名。如下:
select name from exam_result where name like '孙_';
5、查询语文成绩好于英语成绩的同学
在where子句中指明筛选条件为:语文成绩>英语成绩,在select的column列表中指明要查询的列为姓名、语文成绩和英语成绩。如下:
select name chinese, english from exam_result where chinese > english;
6、查询总成绩在200分以下的同学
在select的column列表中添加表达式查询,查询的表达式为语文、数学和英语成绩之和,为了方便观察可以将表达式对应的列指定别名为“总分”,在where子句中指明筛选条件为三科成绩之和小于200。如下:
select name, chinese, math, english, chinese+math+english 总分 from exam_result where 总分 < 200;
结果出现了错误,为什么呢?
因为前面我们说过where子句的执行顺序是在select之前的,而我们是在select时才进行了取别名,所以where子句中的总分字段找不到,因此报错。
而且我们的where子句中也不支持起别名。所以我们只能完整写出:
select name, chinese, math, english , chinese+math+english 总分 from exam_result where chinese+math+english < 200;
注意点:在where子句中不能使用select中指定的别名:
7、语文成绩 > 80 并且不姓孙的同学
在where子句中指明筛选条件为语文成绩大于80,并且and连接下一个条件,再通过模糊匹配和not否定来保证该同学不姓孙,在select的column列表中指明要查询的列为姓名和语文成绩。如下:
select name, chinese from exam_result where chinese > 80 and name not like '孙%';
8、孙某同学,否则要求总成绩 > 200 并且 语文成绩 < 数学成绩 并且 英语成绩 > 80
很显然该题目是一个使用or逻辑运算符的多选题目,要求是查询成绩,被查询的人要么是孙某,要么总成绩大于200分并且语文成绩小于数学成绩并且英语成绩大于80分,查询时需要用到模糊匹配、表达式查询和逻辑运算符。如下:
select name chinese,math,english, chinese + math + english as 总分 from exam_result where name like '孙_' or (chinese+math+english>200 andd chinese < math and english > 80);
9、NULL的查询
在前面我们讲表的创建时我们已经创建了一张学生表,学生表中的内容如下:
select * from students;
这里我们将数据进行一下更新,给孙行者添加一下qq号。
update students set qq='1111111' where name='孙行者';
然后我们再进行查看表格
9.1、 查询 qq 号已知的同学姓名
qq号已知即qq号不为空,所以我们需要在where子句中指明筛选条件为QQ号不为NULL,在select的column列表中指明要查询的列为姓名和QQ号。如下:
select name, qq from students where qq is not null;
9.2、 查询QQ号未知的同学
qq号未知即qq号为空,所以我们需要在where子句中指明筛选条件为QQ号为NULL,在select的column列表中指明要查询的列为姓名和QQ号。如下:
select name, qq from students where qq is null;
当然这里也可以使用<=>来进行判断,但需要注意的是,在与NULL值作比较的时候应该使用<=>运算符,使用=运算符无法得到正确的查询结果。如下:
select name, qq from students where qq <=> null; select name, qq from students where qq <> null;
有些时候我们对我们查找到的数据可能要进行一下排序,于是这里我们就要学习一下MySQL怎么对结果进行排序了。
结果排序的SQL如下:
SELECT ... FROM table_name [WHERE ...] ORDER BY column [ASC | DESC] [, ...];
解释:
注意: 如果查询SQL中没有order by子句,那么返回的顺序是未定义的。
下面我们来进行一些简单的练习:
1、 同学及数学成绩,按数学成绩升序显示
在select的column列表中指明要查询的列为姓名和数学成绩,在order by子句中指明按照数学成绩进行升序排序。如下:
select name, math from exam_result order by math asc;
当然我们也还可以省略order by 最后的asc,默认就是升序,但是这样的写法还是不太建议。
2、查询同学及其QQ号,按QQ号排序显示
这个需求也很简单,我们只需要对qq进行order by asc就行了,但是别忘了我们的students表中的qq有的是null,那么结果会是什么呢?
select name, qq from students order by qq asc;
可以看出在MySQL中默认null是比任何数据都要小的!
select name, qq from students order by qq desc;
结果符合我们的预期!
3、查询同学各门成绩,依次按 数学降序,英语升序,语文升序的方式显示
由于排序中可能存在相同值,于是我们就要再进行比较其他列来确定最后的相对排序了。
排序这里我们不需要使用or来连接多个条件,在order by子句中指明依次按照数学成绩排降序、英语成绩排升序和语文成绩排升序。
在select的column列表中指明要查询的列为姓名、数学成绩、英语成绩和语文成绩,如下:
可以看到显示结果是按照数学成绩进行降序排序的,而相同的数学成绩之间则是按照英语成绩进行升序排序的。
说明:
4、查询同学及总分,由高到低
在select的column列表中指明要查询的列为姓名和总分,在order by子句中指明按照总分进行降序排序。如下:
select name, chinese+ math + english 总分 from exam_result order by 总分 desc;
那么这里的别名为什么我们order by中能够使用呢?
因为只有有了数据才能对数据进行排序,所以order by子句的执行的优先级是非常低的!所以order by能够使用select 中的别名。
也就是说,order by子句的执行是在select语句之后的,所以在order by子句中可以使用别名。
5、查询姓孙的同学或姓曹的同学及其数学成绩,按数学成绩降序显示
由于这里要排序的不是所有数据,所以这里面我们要结合 where 子句 和 order by子句。
这里我们先通过where子句模糊匹配找到人:
select name, math from exam_result where name like '孙%' or name like '曹%';
当查询到目标数据后再在查询SQL后添加order by子句,在order by子句中指明按照数学成绩进行降序排序。如下:
select name, math from exam_result where name like '孙%' or name like '曹%' order by math desc;
有时候我们即使经过了数据的筛选,但是数据依然很多,如果我们直接查看的话是不利于我们进行操作的,所以有时候我们还要对我们的数据进行分页显示。
筛选分页结果的语法如下:
从第0条记录开始,向后筛选出n条记录:
SELECT ... FROM table_name [WHERE ..] [ORDER BY ...] LIMIT n;
从第s条记录开始,向后筛选出n条记录:
SELECT ... FROM table_name [WHERE ...] [ORDER BY ...] LIMIT s, n;
从第s条记录开始,向后筛选出n条记录:
SELECT ... FROM table_name [WHERE ...] [ORDER BY ...] LIMIT n OFFSET s;
说明:
建议: 对未知表进行查询时最好在查询SQL后加上limit 1,避免在查询全表数据时因为表中数据过大而导致数据库卡死。
例、按 id 进行分页,每页 3 条记录,分别显示 第 1、2、3 页
我们先来看一看所有的数据:
select * from exam_result;
查询第1页记录时在查询全表数据的SQL后,加上limit 3子句表示从第0条向后筛选出3条记录。如下:
select * from exam_result limit 3;
现在我们完成了第一页的显示,但是我们要进行第二页的显示时,我们就要指明起始位置为3了,于是我们的limit子句就要变成了limit 3,3。
select * from exam_result limit 3,3;
最后一页,同理我们可以使用limit 6,3,也可以使用 limit 3 offset 6,表明从第6条记录开始,向后筛选出3条记录。如下:
select * from exam_result limit 3 offset 6;
因为从表中筛选出的记录不足3个,所以筛选出几个就显示几个。
修改数据的SQL如下:
UPDATE table_name SET column1=expr1 [, column2=expr2] ... [WHERE ...] [ORDER BY ...] [LIMIT ...];
说明:
1、将孙悟空同学的数学成绩修改为80分
在修改数据之前,我们先查看孙悟空同学当前的数学成绩。如下:
select name, math from exam_result where name='孙悟空';
在update语句中指明要将筛选出来的记录的数学成绩改为80分,并在修改后再次查看数据确保数据成功被修改。如下:
update exam_result set math=80 where name='孙悟空';
2、将曹孟德同学的数学成绩修改为60分,语文成绩修改为70分
同理,在修改数据之前,先查看曹孟德同学当前的数学成绩和语文成绩。如下:
select * from exam_result where name='曹孟德';
在update语句中指明要将筛选出来的记录的数学成绩改为60分,语文成绩改为70分,并在修改后再次查看数据确保数据成功被修改。如下:
select * from exam_result where name='曹孟德';
3、将总成绩倒数前三的3位同学的数学成绩加上30分
在修改数据之前,先查看总成绩倒数前三的3位同学的数学成绩。如下:
select name, chinese+math+english 总分 from exam_result order by 总分 asc; select name, chinese+math+english 总分 from exam_result order by 总分 asc limit 3;
在update语句中指明要将筛选出来的记录的数学成绩加上30分,如下:
update exam_result set math = math + 30 order by chinese+math+english asc limit 3;
并在修改后再次查看数据确保数据成功被修改。
需要注意的是,MySQL中不支持+=这种复合赋值运算符,此外,这里在查看更新后的数据时不能查看总成绩倒数前三的3位同学,因为之前总成绩倒数前三的3位同学,数学成绩加上30分后可能就不再是倒数前三了。
4、将所有同学的语文成绩修改为原来的2倍
在修改数据之前,先查看所有同学的语文成绩。如下:
在update语句中指明要将筛选出来的记录的语文成绩变为原来的2倍,因为要修改的的所有同学所以我们不需要使用where子句,如下:
update exam_result set chinese = chinese * 2 ;
查看修改后的数据:
注意: 这里没有 where子句,会更新全表,更新全表的语句要慎用!
删除数据的SQL如下:
DELETE FROM table_name [WHERE ...] [ORDER BY ...] [LIMIT ...];
说明:
删除表中的某个条目
删除孙悟空同学的考试成绩
在删除数据之前,先查看孙悟空同学的相关信息。
select * from exam_result where name='孙悟空';
然后在delete语句中指明删除孙悟空对应的记录,并在删除后再次查看数据确保数据成功被删除。如下:
delete from exam_result where name='孙悟空';
删除整张表数据
删除整表的操作很简单,我们只要不在delete后面加where子句,这样我们就能够删除整个表了
注意:删除整表操作要慎用!
我们先来准备一张测试表,表中包含一个自增长的主键id和姓名。如下:
create table test( id int primary key auto_increment, name varchar(10)); desc test;
插入数据,并查看数据
insert into test(name) values ('A'),('B'),('C'); select * from test;
这里我们顺便看一下test表的主键值的相关信息,方便我们解释下面的截断表问题:
现在我们在delete语句中只指明要删除数据的表名,而不通过where、order by和limit指明筛选条件,这时将会删除整张表的数据。如下:
这里我们再次查看一下test表的主键值的相关信息:
show create table test \G
我们在插入一些数据进行验证:
insert into test(name) values ('D'), ('E'), ('F'); select * from test;
再次查看主键值的相关信息:
show create table test \G
注意:当通过delete语句删除整表数据时,不会重置AUTO_INCREMENT=n字段!
如果我们想要清除AUTO_INCREMENT=n字段,可以使用截断表!
截断表的SQL语法如下:
TRUNCATE [TABLE] table_name;
说明:
为了演示truncate的效果,我们创建一张测试和原来一样表,表中包含一个自增长的主键id和姓名。如下:
create table for_truncate( id int primary key auto_increment, name varchar(10) ); insert into for_truncate(name) values ('A'),('B'),('C'); select * from for_truncate;
我们查看一下表for_truncate主键值的相关信息:
show create table for_truncate \G
然后我们就要通过truncate删除这张表了,在truncate语句中只指明要删除数据的表名,这时便会删除整张表的数据,但由于truncate实际不对数据操作,因此执行truncate语句后看到影响行数为0。如下:
再次查看表for_truncate主键值的相关信息
再向表中插入一些数据,在插入数据时不指明自增长字段的值,这时会发现插入数据对应的自增长id值是重新从1开始增长的。如下:
insert into for_truncate(name) values ('D'), ('E'), ('F'); select * from for_truncate;
再次查看表for_truncate主键值的相关信息
show create table for_truncate \G
注意: truncate在删除数据时不经过真正的事务,无法回滚,所以截断表操作要慎用!
经过上面的学习,我们对于表的增删查改基本上都没有问题了,但是在这里我们还要学习一下如果将这些SQL组合起来使用。
例如有些时候我们要插入的数据是:某个查询的结果,这时我们就要将SQL组合在一起了。
插入查询结果的SQL语法:
INSERT INTO table_name [(column [, column ...])] SELECT ...
说明:
为了更好的学习这个SQL语法,我们来看一下面的这个案例:
现在有一张表,表中的数据是有重复的,要求如下:
删除表中的的重复复记录,重复的数据只能有一份
我们先来伪造一下题目的数据:
CREATE TABLE duplicate_table (id int, name varchar(20));
INSERT INTO duplicate_table VALUES (100, 'aaa'), (100, 'aaa'), (200, 'bbb'), (200, 'bbb'), (200, 'bbb'), (300, 'ccc');
现在我们查看一下表中的数据:
select * from duplicate_table;
那我们怎么完成题目的要求呢?
思路:
创建一张空表 no_duplicate_table,其结构和 duplicate_table 一样的。
对 duplicate_table表进行查询去重。
将去重的结果插入到新表no_duplicate_table
这里我们就会发现:第二步和第三步的SQL必须要合在一起才能完成我们想要的插入
将表duplicate_table进行重命名为old_duplicate_table
将表no_duplicate_table进行重命名为 duplicate_table。
操作5和操作6的重命名是为了保证底层数据表的改变不影响上层应用的使用
create table no_duplicate_table like duplictae_table; desc no_duplicate_table;
select distinct * from duplicate_table;
insert into no_duplicate_table select distinct * from duplicate_table; select * from no_duplicate_table;
4. 将表duplicate_table重命名为old_duplicate_table
要修改表名,我们使用 RENAME TABLE语句如下:
RENAME TABLE old_table_name TO new_table_name;
旧表( old_table_name)必须存在,而新表( new_table_name)一定不存在,如果新表 new_table_name 确实存在,该语句将失败。
rename table duplicate_table to old_duplicate_table; show tables;
rename table no_duplicate_table to duplicate_table; show tables;