MySQL最著名的EMP练习表:
create table DEPT
(
DEPTNO int(2) not null,--部门编号
DNAME varchar(14),--部门名称
LOC varchar(13)--部门地址
);
alter table DEPT add constraint PK_DEPT primary key (DEPTNO);--设定主键为部门编号
create table EMP
(
EMPNO int(4) not null,--员工编号
ENAME varchar(10),--员工名称
JOB varchar(9),--岗位名称
MGR int(4),--管理人员编号
HIREDATE date,--入职时间
SAL int(7 ),--薪资
COMM int(7 ),--奖金
DEPTNO int(2)--部门编号
);
alter table EMP add constraint PK_EMP primary key (EMPNO);--设定主键为empno
alter table EMP add constraint FK_DEPTNO foreign key (DEPTNO) references DEPT (DEPTNO);
--设定外键关联为depino
insert into DEPT (DEPTNO, DNAME, LOC) values (10, 'ACCOUNTING', 'NEW YORK');--c插入原始数-据
insert into DEPT (DEPTNO, DNAME, LOC) values (20, 'RESEARCH', 'DALLAS');
insert into DEPT (DEPTNO, DNAME, LOC) values (30, 'SALES', 'CHICAGO');
insert into DEPT (DEPTNO, DNAME, LOC) values (40, 'OPERATIONS', 'BOSTON');
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO) values (7369, 'SMITH', 'CLERK', 7902, str_to_date('17-12-1980', '%d-%m-%Y'), 800,null,20);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7499, 'ALLEN', 'SALESMAN', 7698, str_to_date('20-02-1981', '%d-%m-%Y'),1600, 300, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7521, 'WARD', 'SALESMAN', 7698, str_to_date('22-02-1981', '%d-%m-%Y'),1250, 500, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7566, 'JONES', 'MANAGER', 7839, str_to_date('02-04-1981', '%d-%m-%Y'),2975, null, 20);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7654, 'MARTIN', 'SALESMAN', 7698, str_to_date('28-09-1981', '%d-%m-%Y'),1250, 1400, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7698, 'BLAKE', 'MANAGER', 7839, str_to_date('01-05-1981', '%d-%m-%Y'),2850, null, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7782, 'CLARK', 'MANAGER', 7839, str_to_date('09-06-1981', '%d-%m-%Y'),2450, null, 10);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7788, 'SCOTT', 'ANALYST', 7566, str_to_date('19-04-1987', '%d-%m-%Y'),3000,null, 20);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7839, 'KING', 'PRESIDENT', null, str_to_date('17-11-1981', '%d-%m-%Y'),5000,null, 10);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7844, 'TURNER', 'SALESMAN', 7698, str_to_date('08-09-1981', '%d-%m-%Y'),1500, 0, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7876, 'ADAMS', 'CLERK', 7788, str_to_date('23-05-1987', '%d-%m-%Y'),1100,null, 20);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7900, 'JAMES', 'CLERK', 7698, str_to_date('03-12-1981', '%d-%m-%Y'),950,null, 30);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7902, 'FORD', 'ANALYST', 7566, str_to_date('03-12-1981', '%d-%m-%Y'),3000,null, 20);
insert into EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO)values (7934, 'MILLER', 'CLERK', 7782, str_to_date('23-01-1982', '%d-%m-%Y'),1300,null, 10);
create table salgrade (
grade numeric primary key,--薪资等级
losal numeric,--最低薪资
hisal numeric--最高薪资
);
insert into salgrade values (1, 700, 1200);--薪资等级为五
insert into salgrade values (2, 1201, 1400);
insert into salgrade values (3, 1401, 2000);
insert into salgrade values (4, 2001, 3000);
insert into salgrade values (5, 3001, 9999);
一. 多表查询
实际开发中往往数据来自不同的表,所以需要多表查询。本节我们用一个简单的公司管理系统,有三张表EMP,DEPT,SALGRADE来演示如何进行多表查询。
显示雇员名、雇员工资以及所在部门的名字因为上面的数据来自EMP和DEPT表,因此要联合查询:
例如:
select * from emp,dept;
这种将多个表的数据进行组和穷举,我们称为笛卡尔积。
但是我们发现表里面有很多不合法的数据,所以我们需要将数据筛选出合法的数据。
其实我们只要emp表中的deptno = dept表中的deptno字段的记录:
select * from emp,dept where emp.deptno=dept.deptno;
显示部门号为10的部门名,员工名和工资:
select dname,ename,sal from emp,dept where emp.deptno=dept.deptno and emp.deptno=10;
显示各个员工的姓名,工资,及工资级别:
select ename,sal,losal,hisal,grade from emp,salgrade where sal between losal and hisal;
二.自连接
自连接是指在同一张表连接查询.
案例:
显示员工FORD的上级领导的编号和姓名(mgr是员工领导的编号--empno)
子查询 :
使用多表查询(自查询):
select t1.ename,t2.empno from emp t1,emp t2 where t1.ename='FORD' and t1.mgr=t2.empno ;
三.子查询
子查询是指嵌入在其他sql语句中的select语句,也叫嵌套查询.
1.单行子查询
显示SMITH同一部门的员工:
select empno, ename from emp where deptno=(select deptno from emp where ename='SMITH');
2.多行子查询
返回多行记录的子查询:
in关键字
例如 a in { b },将a在b中筛选出来。
查询和10号部门的工作岗位相同的雇员的名字,工岗位,资,部门号,但是不包含10自己的
select ename,job,sal,deptno from emp where job in (select job from emp where deptno=10) and deptno <> 10;
all关键字
all{集合},表示某一个集合中的全部元素。
显示工资比部门30的所有员工的工资高的员工的姓名、工资和部门号:
select ename,sal,deptno from emp where sal > all(select sal from emp where deptno=30);
any关键字
any{集合} 集合中的任意一个。
显示工资比部门30的任意员工的工资高的员工的姓名、工资和部门号(包含自己部门的员工):
select ename,sal,deptno from emp where sal > any(select sal from emp where deptno=30);
3.多列子查询
单行子查询是指子查询只返回单列,单行数据;多行子查询是指返回单列多行数据,都是针对单列而言的,而多列子查询则是指查询返回多个列数据的子查询语句。
案例:查询和SMITH的部门和岗位完全相同的所有雇员,不含SMITH本人
select ename, job,deptno from emp where (deptno,job) = (select deptno, job from emp where ename = 'SMITH') and ename <> 'SMITH';
4.在from子句中使用子查询
子查询语句出现在from子句中。这里要用到数据查询的技巧,把一个子查询当做一个临时表使用。
案例:显示每个高于自己部门平均工资的员工的姓名、部门、工资、平均工资
select ename,deptno,sal,tmp.myavg from emp,(select avg(sal) myavg,deptno dt from emp group by deptno) tmp where sal>myavg and deptno = dt ;
查找每个部门工资最高的人的姓名、工资、部门、最高工资 :
select ename,deptno,sal,maxsal,tmp.dt from emp,(select max(sal) maxsal,deptno dt from emp group by deptno) tmp where deptno=tmp.dt and sal=tmp.maxsal;
显示每个部门的信息(部门名,编号,地址)和人员数量 :
select dept.dname,dept.deptno,dept.loc,total from (select count(*) total ,deptno from emp group by deptno) tmp,dept where tmp.deptno=dept.deptno;
5.合并查询
在实际应用中,为了合并多个select的执行结果,可以使用集合操作符 union,union all
union:
该操作符用于取得两个结果集的并集。当使用该操作符时,会自动去掉结果集中的重复行。
案例:将工资大于2500或职位是MANAGER的人找出来:
union all:
该操作符用于取得两个结果集的并集。当使用该操作符时,不会去掉结果集中的重复行。
select empno,ename,job,sal from emp where sal > 2500 union all select empno,ename,job,sal from emp where job='MANAGER';
大家可以自行尝试,多练多熟悉。
感谢阅读!!!!!!!!