一对多建表原则(都是在多方表中维护):
在多方表中添加一个字段,把该字段作为外键,并且指向一方表中的主键。
将关系的属性及非多方的主标识加入到多方表。
多方表的外键是非多方实体的主标识。
(2)M:N的联系中需要引入关系表
例如:在生活中一个学生可以选择多门课程,每一门课程也可以被多个学生所选择。这些例子在数据库设计中可以使用多对多来完成。
多对对建表原则:
需要创建第三张表,该表中至少有两个字段,作为外键分别指向多对多的双方的主键。
联系的属性及两个实体的主体标识形成关系表。
关系表的主键为两个实体主标识的组合。
(3)1:1的联系中不用引入关系表(很少用)
在生活中,1对1的应用比较少,因为完全可以作为一张表出现。但是有可能公司的业务原因,而需要设计1对1的表结构。例如:一个公司对应一个地址,一个地址也只能对应一个公司。
一对一建表原则:
第一种是主键对应:
将两个表的主键进行关联
第二种是唯一外键对应:
在任何一方加一个外键,但是需要设置成唯一(unique),指向另一方的主键
假设模拟一个购物网站的表结构设计
那购物的网站一般都包含一些基本实体:例如用户、订单、商品、分类等等。
设计表结构首先需要分清楚各个实体之间的关系。
例如用户可以产生多个订单、一种分类下可以有多个商品、一个订单中可以有多个商品、一种商品可以在多个订单中。
设计如下图:
笛卡尔积
表A 表B
aid aname bid bname
a1 aa1 b1 bb1
a2 aa2 b2 bb2
b3 bb3
查询的语法
select * from 表A,表B; 返回的结果就是笛卡尔积。
结果:
a1 aa1 b1 bb1
a1 aa1 b2 bb2
a1 aa1 b3 bb3
a2 aa2 b1 bb1
a2 aa2 b2 bb2
a2 aa2 b3 bb3
select * from dept,emp;
先导入SQL表
DROP TABLE dept; --部门表 CREATE TABLE dept( deptno int PRIMARY KEY, dname VARCHAR(14) , --部门名称 loc VARCHAR(13) ---部门地址 ) ; CREATE TABLE dept( deptno int PRIMARY KEY, dname VARCHAR(14) , loc VARCHAR(13) ) ; INSERT INTO DEPT VALUES (10,'ACCOUNTING','NEW YORK'); INSERT INTO DEPT VALUES (20,'RESEARCH','DALLAS'); INSERT INTO DEPT VALUES (30,'SALES','CHICAGO'); INSERT INTO DEPT VALUES (40,'OPERATIONS','BOSTON'); drop table emp; --员工表 CREATE TABLE emp( empno int PRIMARY KEY, --员工编号 ename VARCHAR(10), ---员工姓名 job VARCHAR(9), --员工工作 mgr int, ----员工直属领导编号 hiredate DATE, ----入职时间 sal double, ---工资 comm double, --奖金 deptno int REFERENCES dept); --关联dept表 CREATE TABLE emp( empno int PRIMARY KEY, ename VARCHAR(10), job VARCHAR(9), mgr int, hiredate DATE, sal double, comm double, deptno int , foreign key emp(deptno) REFERENCES dept(deptno) ); INSERT INTO EMP VALUES(7369,'SMITH','CLERK',7902,"1980-12-17",800,NULL,20); INSERT INTO EMP VALUES(7499,'ALLEN','SALESMAN',7698,'1981-02-20',1600,300,30); INSERT INTO EMP VALUES(7521,'WARD','SALESMAN',7698,'1981-02-22',1250,500,30); INSERT INTO EMP VALUES(7566,'JONES','MANAGER',7839,'1981-04-02',2975,NULL,20); INSERT INTO EMP VALUES(7654,'MARTIN','SALESMAN',7698,'1981-09-28',1250,1400,30); INSERT INTO EMP VALUES(7698,'BLAKE','MANAGER',7839,'1981-05-01',2850,NULL,30); INSERT INTO EMP VALUES(7782,'CLARK','MANAGER',7839,'1981-06-09',2450,NULL,10); INSERT INTO EMP VALUES(7788,'SCOTT','ANALYST',7566,'1987-07-03',3000,NULL,20); INSERT INTO EMP VALUES(7839,'KING','PRESIDENT',NULL,'1981-11-17',5000,NULL,10); INSERT INTO EMP VALUES(7844,'TURNER','SALESMAN',7698,'1981-09-08',1500,0,30); INSERT INTO EMP VALUES(7876,'ADAMS','CLERK',7788,'1987-07-13',1100,NULL,20); INSERT INTO EMP VALUES(7900,'JAMES','CLERK',7698,'1981-12-03',950,NULL,30); INSERT INTO EMP VALUES(7902,'FORD','ANALYST',7566,'1981-12-03',3000,NULL,20); INSERT INTO EMP VALUES(7934,'MILLER','CLERK',7782,'1981-01-23',1300,NULL,10); CREATE TABLE salgrade( grade int,--等级 losal double, --最低工资 hisal double ); --最高工次 CREATE TABLE salgrade( grade int, losal double, hisal double ); 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);
练习:
(1)查看XXX所属的部门名称和员工名称?
select d.deptno,d.dname,e.ename from dept d,emp e where d.deptno = e.deptno ;
(2)统计每个部门的人数(按照部门名称统计)
select d.deptno,d.dname,count(*) from dept d,emp e where d.deptno = e.deptno group by d.deptno order by d.deptno asc;
(3)统计部门的平均工资(按部门名称统计)
select d.dname,avg(sal) from dept d,emp e where d.deptno = e.deptno group by d.dname;
(4)统计部门的平均工资大于公司平均工资的部门
select d.dname,avg(sal) as sa from dept d,emp e where d.deptno = e.deptno group by d.dname having sa > (select avg(sal) from emp );
基本查询练习:
1、查找部门30中员工的详细信息。 select * from emp where deptno=30; 2、找出从事clerk工作的员工的编号、姓名、部门号。 select empno,ename,deptno from emp where job='clerk'; 3、检索出奖金多于基本工资的员工信息。 select * from emp where comm >sal 4、检索出奖金多于基本工资60%的员工信息。 select * from emp where comm>sal*0.6; 5、找出10部门的经理、20部门的职员 的员工信息。 select * from emp where deptno=10 and job='MANAGER' or deptno=20 and job='CLERK' 6、找出10部门的经理、20部门的职员 或者既不是经理也不是职员但是工资高于2000元的员工信息。 select * from emp where (deptno=10 and job='MANAGER') or ( deptno=20 and job='CLERK') or ( job not in ('MANAGER','CLERK') and sal>2000) 7、找出获得奖金的员工的工作。 select * from emp where comm is not null; 8、找出奖金少于100或者没有获得奖金的员工的信息。 select * from emp where comm is null or comm<100; 9、找出姓名以A、B、S开始的员工信息。 select * from emp where ename like 'A%' OR ENAME LIKE 'B%' OR ENAME LIKE 'S%'; 10、找到名字长度为7个字符的员工信息。 select * from emp where length(ename)=7; select * from emp where ename like '______'; 11、名字中不包含R字符的员工信息。 select * from emp where ename not like '%R%'; 12、返回员工的详细信息并按姓名排序。 select * from emp order by ename; 13、返回员工的信息并按工作降序工资升序排列。 select * from emp order by job desc,sal asc; 14、计算员工的日薪(按30天)。 select sal/30 from emp 15、找出姓名中包含A的员工信息。 select * from emp where ename like '%A%';多表查询与分组练习
//多表与分组练习 1、返回拥有员工的部门名、部门号。 select d.deptno,d.dname from dept d,emp e where d.deptno = e.deptno group by d.deptno; 2、工资水平多于smith的员工信息。 select * from emp e where e.sal > (select em.sal from emp em where em.ename = 'smith' ); 3、返回员工和所属经理的姓名。 select e.ename,em.ename from emp e,emp em where e.mgr = em.empno and em.job = 'MANAGER'; 4、返回雇员的雇佣日期早于其经理雇佣日期的员工及其经理姓名。 select e.ename,em.ename from emp e,emp em where e.mgr = em.empno and em.job = 'MANAGER' and e.hiredate < em.hiredate; 5、返回员工姓名及其所在的部门名称。 select d.dname,e.ename from dept d,emp e where d.deptno = e.deptno; 6、返回从事clerk工作的员工姓名和所在部门名称。 select d.dname,e.ename,e.job from dept d,emp e where d.deptno = e.deptno and job = 'clerk'; 7、返回部门号及其本部门的最低工资。 select d.deptno,min(e.sal) from dept d,emp e where d.deptno = e.deptno group by d.deptno; 8、返回销售部(sales)所有员工的姓名。 select * from dept d,emp e where d.deptno = e.deptno and d.dname = 'sales'; 9、返回工资水平多于平均工资的员工。 select * from emp e where e.sal > (select avg(sal) from emp) 10、返回与SCOTT从事相同工作的员工。 select * from emp e where e.job = (select job from emp em where em.ename = 'SCOTT'); 11、返回与30部门员工工资水平相同的员工姓名与工资。 select * from emp e where e.sal in (select e.sal from dept d,emp e where d.deptno = e.deptno and d.deptno = 30); 12、返回工资高于30部门所有员工工资水平的员工信息。 select * from emp e where e.sal > (select avg(e.sal) from dept d,emp e where d.deptno = e.deptno and d.deptno = 30); 13、返回部门号、部门名、部门所在位置及其每个部门的员工总数。 select d.deptno,d.dname,d.loc,count(*) as total from dept d,emp e where d.deptno = e.deptno group by d.deptno; 14、返回员工的姓名、所在部门名及其工资。 select e.ename,d.dname,e.sal from dept d,emp e where d.deptno = e.deptno; 15、返回员工的详细信息。(包括部门名) select * from dept d,emp e where d.deptno = e.deptno; 16、返回员工工作及其从事此工作的最低工资。 select e.job,min(e.sal) from dept d,emp e where d.deptno = e.deptno group by e.job; 17、返回不同部门经理的最低工资。 select min(e.sal) from dept d,emp e where d.deptno = e.deptno and e.job = 'MANAGER' group by d.deptno ; 18、计算出员工的年薪,并且以年薪排序。 select e.sal*12 from emp e order by e.sal*12 desc; 19、返回工资处于第四级别的员工的姓名。 select e.ename from emp e where e.sal > (select losal from salgrade where grade = 4) and e.sal < (select hisal from salgrade where grade = 4); 20、返回工资为二等级的职员名字、部门所在地、和二等级的最低工资和最高工资 select e.ename,d.loc,s.losal,s.hisal from dept d,emp e,salgrade s where d.deptno = e.deptno and e.sal between s.losal and s.hisal and s.grade = 2;