03 表的操作以及单表查询
1. 修改、删除被关联字段
主表(父表)、从表(子表)
show create table 从表; # 查看外键名称
create table t1(id int, pid int constraint 名字 foreign key(关联字段) reference 主表(被关联字段) ;
# 创建外键时可以指定外键名称(不能创建后修改)
查看所有外键的名称的方法:
select REFERENCED_TABLE_SCHEMA,REFERENCED_TABLE_NAME,
REFERENCED_COLUMN_NAME,table_name,CONSTRAINT_NAME from information_schema.key_column_usage;
#包含我们创建外键的时候,mysql帮我们自动生成的外键名称。
删除外键关系
alter table 从表 drop foreign key 外键名称;
删除主表关联字段
alter table 主表 drop 关联字段名称; # 删除外键关系后,才能删关联的字段
添加字段
alter table 表名 add 字段名称 数据类型 约束条件;
创建表后,再添加外键关系
alter table 从表 add foreign key(关联字段) reference 主表(被关联字段) ;
2. 级联
级联的模式:
严格模式(district),外键有强制约束效果,被关联字段不能随意删除和修改。
级联模式(cascade),外键有强制约束效果,被关联字段删除或者修改,关联字段的数据也会删除或者修改。
置空模式(set null),被关联字段删除时,关联它的字段数据会置换成null,不会删除。
外键约束有三种约束模式(都是针对父表的约束):
模式一: district 严格约束(默认的 ),父表不能删除或者更新已经被子表数据引用的记录
模式二:cascade 级联模式:父表的操作,对应的子表关联的数据也跟着操作 。
模式三:set null:置空模式,父表操作之后,子表对应的数据(外键字段)也跟着被置空。
通常的一个合理的约束模式是:删除的时候子表置空;更新的时候子表级联。
指定模式的语法:foreign key(外键字段) references 父表(主键字段)on delete 模式 on update 模式;
注意:删除置空的前提条件是 外键字段允许为空,不然外键会创建失败。
外键虽然很强大,能够进行各种约束,但是外键的约束降低了数据的可控性和可拓展性。通常在实际开发时,很少使用外键来约束
alter table 从表 add foreign key(从表字段) reference 主表(主表字段) on delete cascade;
# 当删除主表数据的时候,从表中有关的数据都跟着删除
alter table 从表 add foreign key(从表字段) reference 主表(主表字段)on update cascade;
# 当主表的关系字段修改的时候,从表对应的关系字段的值也更着更新。
# 例如
constraint fk_t1_publish foreign key(pid) references publish(id) on delete cascade on update cascade;
3. 表详细操作
alter table 表名 + 操作
# 1.修改表名
alter table 表名 rename 新表名;
# 2.增加字段
alter table 表名 add 字段名 数据类型 约束条件
first; #添加字段,并将其放在第一个字段的前面(开头)
after; # after表示放在最后字段的后面(末尾)
#通过一个first和一个after就可以将新添加的字段放到表的任意字段位置。
# 3.删除字段
alter table 表名 drop 字段名;
# 4.修改字段
alter table 表名 modify 字段名 新数据类型 约束条件;
# 修改字段数据类型
alter table 表名 change 旧字段名 新字段名 新数据类型 约束条件;
# 既可以修改数据类型(也可不更改),又可以修改字段名字
4. 行记录操作
4.1 增加 insert
1. 插入完整数据(顺序插入)
语法一:
insert into 表名(字段1,字段2,字段3…字段n) values(值1,值2,值3…值n); #指定字段来插入数据,插入的值要和你前面的字段相匹配
语法二:
insert into 表名 values (值1,值2,值3…值n); #不指定字段的话,就按照默认的几个字段来插入数据
2. 指定字段插入数据
语法:
insert into 表名(字段1,字段2,字段3…) values (值1,值2,值3…);
3. 插入多条记录
语法:#插入多条记录用逗号来分隔
insert into 表名 values
(值1,值2,值3…值n),
(值1,值2,值3…值n),
(值1,值2,值3…值n);
4. 插入查询结果
语法:
insert into 表名(字段1,字段2,字段3…字段n)
select (字段1,字段2,字段3…字段n) from 表2
where …; #将从表2里面查询出来的结果来插入到我们的表中,但是注意查询出来的数据要和我们前面指定的字段要对应好
4.2 修改(更新)update
update 表名 set 字段1=值1,字段2=值2 where 条件;
4.3 删除 delete
delete from 表名 where 条件; # 删除指定条件字段的值
delete from 表名; # 删除所有的数据,但是不会重置自增字段的数字
truncate 表名; # 全清空,可使自增字段重置
4.4 查询 select
分单表查询与多变查询
5. 单表查询
#查询数据的本质:mysql会到你本地的硬盘上找到对应的文件,然后打开文件,按照你的查询条件来找出你需要的数据。下面是完整的一个单表查询的语法
select * from,这个select * 指的是要查询所有字段的数据。
select distinct 字段1,字段2... from 库名.表名 #from后面是说从库的某个表中去找数据,mysql会去找到这个库对应的文件夹下去找到你表名对应的那个数据文件,找不到就直接报错了,找到了就继续后面的操作
where 条件; #从表中找符合条件的数据记录,where后面跟的是你的查询条件
group by field(字段) #分组
having 筛选 #过滤,过滤之后执行select后面的字段筛选,就是说我要确定一下需要哪个字段的数据,你查询的字段数据进行去重,然后在进行下面的操作
order by field(字段) #将结果按照后面的字段进行排序
limit 限制条数 #将最后的结果加一个限制条数,就是说我要过滤或者说限制查询出来的数据记录的条数
5.1 关键字的执行优先级(重点)
重点中的重点:关键字的执行优先级
from
where
group by
having
select
distinct
order by
limit
步骤:
1.找到表:from
2.拿着where指定的约束条件,去文件/表中取出一条条记录
3.将取出的一条条记录进行分组group by,如果没有group by,则整体作为一组
4.将分组的结果进行having过滤
5.执行select
6.去重
7.将结果按条件排序:order by
8.限制结果的显示条数
5.2 查询
select * from 表名; # 查看全部,不推荐*,效率低
select 字段1,字段2 from 表名;
select distinct 字段 from 表名; # 对查询出来的记录进行去重,如果对多个字段进行去重,会对数据重复少的字段去重
# distinct 前面不允许写字段
# 四则运算
select 字段运算 from 表名; # 结果显示:运算字段的显示名字会变成运算表达式,例如:select salary*12 from empioyee; 字段名会显示 salary*12,不美观,
可以用 as + 新字段名 来指定。
# 自定义格式 字节规定查询结果的显示格式 concat()
select concat("姓名:",name, "年薪:",salary*12) as Salary from empioyee;
# 拼接字符串,放在一列内。
5.3 where约束
where语句中可以使用:
- 比较运算符:> 、< 、>=、 <= 、<> (不等于)、!=
- between 80 and 100 # 值在80到100之间
- in(80,90,100) # 值是80或90或100
- like 'egon%'
pattern可以是%或_,
%表示匹配任意多字符
_表示匹配一个字符 可多写几个_ - 逻辑运算符:在多个条件直接可以使用逻辑运算符 and、or、not
#1:单条件查询
SELECT name FROM employee
WHERE post='sale'; #注意优先级,我们说where的优先级是不是比select要高啊,所以我们的顺序是先找到这个employee表,然后按照post='sale'的条件,然后去表里面select数据
#2:多条件查询
SELECT name,salary FROM employee
WHERE post='teacher' AND salary>10000;
#3:关键字BETWEEN AND 写的是一个区间
SELECT name,salary FROM employee
WHERE salary BETWEEN 10000 AND 20000; #就是salary>=10000 and salary<=20000的数据
SELECT name,salary FROM employee
WHERE salary NOT BETWEEN 10000 AND 20000; #加个not,就是不在这个区间内,薪资小于10000的或者薪资大于20000的,注意没有等于,
#4:关键字IS NULL(判断某个字段是否为NULL不能用等号,需要用IS) 判断null只能用is
SELECT name,post_comment FROM employee
WHERE post_comment IS NULL;
SELECT name,post_comment FROM employee
WHERE post_comment IS NOT NULL;
SELECT name,post_comment FROM employee
WHERE post_comment=''; 注意''是空字符串,不是null,两个是不同的东西,null是啥也没有,''是空的字符串的意思,是一种数据类型,null是另外一种数据类型
ps:
执行
update employee set post_comment='' where id=2;
再用上条查看,就会有结果了
#5:关键字IN集合查询
SELECT name,salary FROM employee
WHERE salary=3000 OR salary=3500 OR salary=4000 OR salary=9000 ; #这样写是不是太麻烦了,写一大堆的or,下面我们用in这个简单的写法来搞
SELECT name,salary FROM employee
WHERE salary IN (3000,3500,4000,9000) ;
SELECT name,salary FROM employee
WHERE salary NOT IN (3000,3500,4000,9000) ;
#6:关键字LIKE模糊查询,模糊匹配,可以结合通配符来使用
通配符’%’ #匹配任意所有字符
SELECT * FROM employee
WHERE name LIKE 'eg%';
通配符’_’ #匹配任意一个字符
SELECT * FROM employee
WHERE name LIKE 'al__'; #注意我这里写的两个_,用1个的话,匹配不到alex,因为al后面还有两个字符ex。
5.4 分组查询:group by
分组发生在where之后,即分组是基于where之后得到的记录而进行的。
分组指的是:将所有记录按照某个相同字段进行归类,比如针对员工信息表的职位分组,或者按照性别进行分组等。(字段是分组依据)
select * from 表名 group by 字段; #默认取分组后的每组第一条数据
# group by关键字与group_concat函数一起使用,可以显示组内某个字段的所有数据
select post,group_concat(name) from employee group by post; #例如,按照岗位分组,并查看组内所有成员名,通过逗号拼接在一起
# 聚合函数
select count(字段),其他字段 from 表名 group by 分组字段;
# 统计分组后,每组个数
聚合函数:count、max、min、avg、sum等。
group by分组时后可以跟多个条件,那么这多个条件同时重复才算一组,用逗号隔开。
ONLY_FULL_GROUP_BY 模式下 :限制查询字段必须是分组依据所用的字段和聚合函数。(你select后面取的字段必须在你的group by后面的字段里面才行。)
5.5 having 分组后过滤
where优先级在group by 前面,因此分组后加条件需havind;
havind后面可以加聚合函数,where不行
select 字段,聚合函数(字段) from 表名 group by 字段 having 聚合函数
having与where区别:
having的语法格式和where是一模一样的,只不过having是在分组之后进行的进一步的过滤,where不能使用聚合函数,having是可以使用聚合函数的。
#!!!执行优先级从高到低:where > group by > having
#1. Where 发生在分组group by之前,因而Where中可以有任意字段,但是绝对不能使用聚合函数。
#2. Having发生在分组group by之后,因而Having中可以使用分组的字段,无法直接取到其他字段。
5.6 去重distinct
1.将查询的结果进行去重:select distinct post from employee;
2.用distinct来返回不重复字段的条数(count(distinct id)),其原因是distinct只能返回他的目标字段,而无法返回其他字段,distinct 想写在其他字段后面需要配合聚合函数来写,否则只会取到第一条记录。
mysql> select count(distinct post) from employee;
+----------------------+
| count(distinct post) |
+----------------------+
| 4 |
+----------------------+
1 row in set (0.00 sec)
5.7 查询排序 order by
默认升序,加 desc是降序。
select * from 表名 order by 字段 asc; #按照字段升序,asc可不写
select * from 表名 order by 字段 desc; #降序
# 多条件排序
select * from 表名 order by 字段1 asc, 字段2 desc;
# 按字段1升序排列,字段1中相同的数据,按照字段2降序排序
5.8 限制结果的显示条数 limit
网站的分页显示功能。默认初始位置为0。
select * from 表名 order by 字段 desc limit 0,5;
# 从第0开始查询,共5条
select * from 表名 order by 字段 desc limit 5,5;
# 从第5条开始查询,共5条
5.9 正则表达式查询
select * from 表名 where 字段 regexp '^a'; # ^a 是查询字段中所有a开头的数据
查看所有员工中名字是jin开头,n或者g结束的员工信息
select * from employee where name regexp '^jin.*[g|n]$';
补充:级联set null的用法和示例
mysql> create table tt2(id int primary key auto_increment,name char(10));
mysql> create table tt3(id int,pid int,foreign key(pid) references tt2(id) on delete set null);
Query OK, 0 rows affected (1.06 sec)
mysql> desc tt3;
+-------+---------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+---------+------+-----+---------+-------+
| id | int(11) | YES | | NULL | |
| pid | int(11) | YES | MUL | NULL | |
+-------+---------+------+-----+---------+-------+
2 rows in set (0.01 sec)
mysql> insert into tt2(name) values('xx1'),('xx2');
Query OK, 2 rows affected (0.14 sec)
Records: 2 Duplicates: 0 Warnings: 0
mysql> insert into tt3 values(1,1),(2,1);
Query OK, 2 rows affected (0.12 sec)
Records: 2 Duplicates: 0 Warnings: 0
mysql> select * from tt3;
+------+------+
| id | pid |
+------+------+
| 1 | 1 |
| 2 | 1 |
+------+------+
2 rows in set (0.00 sec)
mysql> delete from tt2 where id = 1;
Query OK, 1 row affected (0.10 sec)
mysql> select * from tt3;
+------+------+
| id | pid |
+------+------+
| 1 | NULL |
| 2 | NULL |
+------+------+
2 rows in set (0.00 sec)