mysql表间关系与表连接查询

mysql学习笔记3

表间关系:一对一,一对多,多对多

一对一:直接将对方主键设置为自身的一个外键(至于哪个表设无关紧要) 例如相对使用比较少。HR系统中,员工与简历(实际开发用的很少,因为一对一完全可以建到一张表里)

一对多:将一作为主表 ,多作为附表, 附表内加一个一的外键, 例如最常用的关系 部门和员工,将部门作为主表,员工表内加一个部门关联的外键.

多对多:再多建一个中间关系表来描述左右表关系,* 例如 选课系统中 课程和学生的关系, 一门课程可以有多个学生选择,一个学生选择多门课程

表连接查询:

三类:交叉查询,内连接,外连接

注:mysql内交叉和内连接结果没什么区别

交叉查询:

内连接查询:
隐式内连接: 关键词select from where
-- 设置过滤条件 Column 'id' in where clause is ambiguous
 select * from emp,dept where id=5; 
 select * from emp,dept where emp.`dept_id` = dept.`id`;

显示内连接:关键词select 显示 form 表1 inner join 表2 on 条件

确定查询哪些表
select * from emp inner join dept;

确定表连接条件,员工表.dept_id = 部门表.id 的数据才是有效的
select * from emp e inner join dept d on e.`dept_id` = d.`id`; 

确定查询条件,我们查询的是唐僧的信息,员工表.name='唐僧'
select * from emp e inner join dept d on e.`dept_id` = d.`id` where e.`name`='唐僧 ';

确定查询字段,查询唐僧的信息,显示员工 id,姓名,性别,工资和所在的部门名称
select e.`id`,e.`name`,e.`gender`,e.`salary`,d.`name` from emp e inner join dept d on e.`dept_id` = d.`id` where e.`name`='唐僧';

我们发现写表名有点长,可以给表取别名,显示的字段名也使用别名
select e.`id` 编号,e.`name` 姓名,e.`gender` 性别,e.`salary` 工资,d.`name` 部门名字 from emp e inner join dept d on e.`dept_id` = d.`id` where e.`name`='唐僧';
外连接:
左外连接:SELECT 字段名 FROM 左表 LEFT [OUTER] JOIN 右表 ON 条件用左边表的记录去匹配右边表的记录,如果符合条件的则显示;否则,显示 NULL
-- 在部门表中增加一个销售部 
insert into dept (name) values ('销售部'); select * from dept; 
 
-- 使用内连接查询 
select * from dept d inner join emp e on d.`id` = e.`dept_id`; 
 
-- 使用左外连接查询 
select * from dept d left join emp e on d.`id` = e.`dept_id`; 
右外连接:SELECT 字段名 FROM 左表 RIGHT [OUTER] JOIN 右表 ON 条件用左边表的记录去匹配右边表的记录,如果符合条件的则显示;否则,显示 NULL 可以理解为:在内连接的基础上保证右表的数据全部显示
-- 在员工表中增加一个员工  
insert into emp values (null, '沙僧','男',6666,'2013-12-05',null); 
select * from emp;  
-- 使用内连接查询  
select * from dept inner join emp on dept.`id` = emp.`dept_id`; 
 
-- 使用右外连接查询 
select * from dept right join emp on dept.`id` = emp.`dept_id`;

子查询:本质就是嵌套查询

-- 1) 查询最高工资是多少 
select max(salary) from emp;
-- 2) 根据最高工资到员工表查询到对应的员工信息 (查询结果为单个时用=号或者in都可)
select * from emp where salary = (select max(salary) from emp);

-- 1) 查询平均工资是多少 
select avg(salary) from emp; 
-- 2) 到员工表查询小于平均的员工信息 
select * from emp where salary < (select avg(salary) from emp); 

-- 先查询大于5000的员工所在的部门id (查询结果为多个时,用in)
select dept_id from emp where salary > 5000; 
-- 再查询在这些部门id中部门的名字   Subquery returns more than 1 row 
select name from dept where id = (select dept_id from emp where salary > 5000); 
select name from dept where id in (select dept_id from emp where salary > 5000)

子查询也可以放在前面select那边
-- 查询出2011年以后入职的员工信息,包括部门名称 
-- 在员工表中查询2011-1-1以后入职的员工 
select * from emp where join_date >='2011-1-1'; 
 
-- 查询所有的部门信息,与上面的虚拟表中的信息组合,找出所有部门id等于的dept_id 
select * from dept d, (select * from emp where join_date >='2011-1-1') e where d.`id`= e.dept_id ;


还可以使用表连接(on 或者  union用来去重)
select * from emp inner join dept on emp.`dept_id` = dept.`id` where join_date >='2011-1-1'; 
select * from emp inner join dept on emp.`dept_id` = dept.`id` and join_date >='2011-1-1'; 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值