039_MySQL_多表查詢

#創建部門
CREATE TABLE IF NOT EXISTS dept (did int not null auto_increment PRIMARY KEY,dname VARCHAR(50) not null COMMENT '部門名稱'
)ENGINE=INNODB DEFAULT charset utf8;#添加部門數據
INSERT INTO `dept` VALUES ('1', '教學部');
INSERT INTO `dept` VALUES ('2', '銷售部');
INSERT INTO `dept` VALUES ('3', '市場部');
INSERT INTO `dept` VALUES ('4', '人事部');
INSERT INTO `dept` VALUES ('5', '鼓勵部');-- 創建人員
DROP TABLE IF EXISTS `person`;
CREATE TABLE `person` (`id` int(11) NOT NULL AUTO_INCREMENT,`name` varchar(50) NOT NULL,`age` tinyint(4) DEFAULT '0',`sex` enum('','','人妖') NOT NULL DEFAULT '人妖',`salary` decimal(10,2) NOT NULL DEFAULT '250.00',`hire_date` date NOT NULL,`dept_id` int(11) DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=13 DEFAULT CHARSET=utf8;-- 添加人員數據-- 教學部
INSERT INTO `person` VALUES ('1', 'alex', '28', '人妖', '53000.00', '2010-06-21', '1');
INSERT INTO `person` VALUES ('2', 'wupeiqi', '23', '', '8000.00', '2011-02-21', '1');
INSERT INTO `person` VALUES ('3', 'egon', '30', '', '6500.00', '2015-06-21', '1');
INSERT INTO `person` VALUES ('4', 'jingnvshen', '18', '', '6680.00', '2014-06-21', '1');-- 銷售部
INSERT INTO `person` VALUES ('5', '歪歪', '20', '', '3000.00', '2015-02-21', '2');
INSERT INTO `person` VALUES ('6', '星星', '20', '', '2000.00', '2018-01-30', '2');
INSERT INTO `person` VALUES ('7', '格格', '20', '', '2000.00', '2018-02-27', '2');
INSERT INTO `person` VALUES ('8', '周周', '20', '', '2000.00', '2015-06-21', '2');-- 市場部
INSERT INTO `person` VALUES ('9', '月月', '21', '', '4000.00', '2014-07-21', '3');
INSERT INTO `person` VALUES ('10', '安琪', '22', '', '4000.00', '2015-07-15', '3');-- 人事部
INSERT INTO `person` VALUES ('11', '周明月', '17', '', '5000.00', '2014-06-21', '4');-- 鼓勵部
INSERT INTO `person` VALUES ('12', '蒼老師', '33', '', '1000000.00', '2018-02-21', null);
創建表和數據

一,多表聯合查詢

#多表查詢語法
select  字段1,字段2... from 表1,表2... [where 條件]

注意: 如果不加條件直接進行查詢,則會出現以下效果,這種結果我們稱之為 笛卡爾乘積

#查詢人員和部門所有信息
select * from person,dept

笛卡爾乘積公式 : A表中數據條數? ?*? B表中數據條數? = 笛卡爾乘積.

mysql> select * from person ,dept;
+----+----------+-----+-----+--------+------+-----+--------+
| id | name     | age | sex | salary | did  | did | dname  |
+----+----------+-----+-----+--------+------+-----+--------+
|  1 | alex     |  28 ||  53000 |    1 |   1 | python |
|  1 | alex     |  28 ||  53000 |    1 |   2 | linux  |
|  1 | alex     |  28 ||  53000 |    1 |   3 | 明教   |
|  2 | wupeiqi  |  23 ||  29000 |    1 |   1 | python |
|  2 | wupeiqi  |  23 ||  29000 |    1 |   2 | linux  |
|  2 | wupeiqi  |  23 ||  29000 |    1 |   3 | 明教   |
|  3 | egon     |  30 ||  27000 |    1 |   1 | python |
|  3 | egon     |  30 ||  27000 |    1 |   2 | linux  |
|  3 | egon     |  30 ||  27000 |    1 |   3 | 明教   |
|  4 | oldboy   |  22 ||      1 |    2 |   1 | python |
|  4 | oldboy   |  22 ||      1 |    2 |   2 | linux  |
|  4 | oldboy   |  22 ||      1 |    2 |   3 | 明教   |
|  5 | jinxin   |  33 ||  28888 |    1 |   1 | python |
|  5 | jinxin   |  33 ||  28888 |    1 |   2 | linux  |
|  5 | jinxin   |  33 ||  28888 |    1 |   3 | 明教   |
|  6 | 張無忌   |  20 ||   8000 |    3 |   1 | python |
|  6 | 張無忌   |  20 ||   8000 |    3 |   2 | linux  |
|  6 | 張無忌   |  20 ||   8000 |    3 |   3 | 明教   |
|  7 | 令狐沖   |  22 ||   6500 | NULL |   1 | python |
|  7 | 令狐沖   |  22 ||   6500 | NULL |   2 | linux  |
|  7 | 令狐沖   |  22 ||   6500 | NULL |   3 | 明教   |
|  8 | 東方不敗 |  23 ||  18000 | NULL |   1 | python |
|  8 | 東方不敗 |  23 ||  18000 | NULL |   2 | linux  |
|  8 | 東方不敗 |  23 ||  18000 | NULL |   3 | 明教   |
+----+----------+-----+-----+--------+------+-----+--------+
現象

注意: 多表查詢時,一定要找到兩個表中相互關聯的字段,并且作為條件使用

#查詢人員和部門所有信息
select * from person,dept where person.did = dept.did;
mysql> select * from person,dept where person.did = dept.did;
+----+---------+-----+-----+--------+-----+-----+--------+
| id | name    | age | sex | salary | did | did | dname  |
+----+---------+-----+-----+--------+-----+-----+--------+
|  1 | alex    |  28 ||  53000 |   1 |   1 | python |
|  2 | wupeiqi |  23 ||  29000 |   1 |   1 | python |
|  3 | egon    |  30 ||  27000 |   1 |   1 | python |
|  4 | oldboy  |  22 ||      1 |   2 |   2 | linux  |
|  5 | jinxin  |  33 ||  28888 |   1 |   1 | python |
|  6 | 張無忌  |  20 ||   8000 |   3 |   3 | 明教   |
|  7 | 令狐沖  |  22 ||   6500 |   2 |   2 | linux  |
+----+---------+-----+-----+--------+-----+-----+--------+
7 rows in set

二,多表連接查詢

#多表連接查詢語法(重點)
select 字段列表from 表1  inner|left|right JOIN  表2
on 表1.字段 = 表2.字段;

1,內連接查詢(只顯示符合條件的數據)

#查詢人員和部門所有信息
select * from person inner join dept  on person.did =dept.did;

  內連接查詢與多表聯合查詢的效果是一樣的

mysql> select * from person inner join  dept  on  person.did =dept.did;
+----+---------+-----+-----+--------+-----+-----+--------+
| id | name    | age | sex | salary | did | did | dname  |
+----+---------+-----+-----+--------+-----+-----+--------+
|  1 | alex    |  28 ||  53000 |   1 |   1 | python |
|  2 | wupeiqi |  23 ||  29000 |   1 |   1 | python |
|  3 | egon    |  30 ||  27000 |   1 |   1 | python |
|  4 | oldboy  |  22 ||      1 |   2 |   2 | linux  |
|  5 | jinxin  |  33 ||  28888 |   1 |   1 | python |
|  6 | 張無忌  |  20 ||   8000 |   3 |   3 | 明教   |
|  7 | 令狐沖  |  22 ||   6500 |   2 |   2 | linux  |
+----+---------+-----+-----+--------+-----+-----+--------+
7 rows in set
View Code

2,左外連接查詢(左邊表中的數據優先全部顯示)

#查詢人員和部門所有信息
select * from person left join  dept  on  person.did =dept.did;

  效果:人員表中的數據全部都顯示,而 部門表中的數據符合條件的才會顯示,不符合條件的會以 null 進行填充.

mysql> select * from person left join  dept  on  person.did =dept.did;
+----+----------+-----+-----+--------+------+------+--------+
| id | name     | age | sex | salary | did  | did  | dname  |
+----+----------+-----+-----+--------+------+------+--------+
|  1 | alex     |  28 ||  53000 |    1 |    1 | python |
|  2 | wupeiqi  |  23 ||  29000 |    1 |    1 | python |
|  3 | egon     |  30 ||  27000 |    1 |    1 | python |
|  5 | jinxin   |  33 ||  28888 |    1 |    1 | python |
|  4 | oldboy   |  22 ||      1 |    2 |    2 | linux  |
|  7 | 令狐沖   |  22 ||   6500 |    2 |    2 | linux  |
|  6 | 張無忌   |  20 ||   8000 |    3 |    3 | 明教   |
|  8 | 東方不敗 |  23 ||  18000 | NULL | NULL | NULL   |
+----+----------+-----+-----+--------+------+------+--------+
8 rows in set
View Code

3,右外連接查詢 (右邊表中的數據優先全部顯示)

#查詢人員和部門所有信息
select * from person right join  dept  on  person.did =dept.did;

  效果:正好與[左外連接相反]

mysql> select * from person right join  dept  on  person.did =dept.did;
+----+---------+-----+-----+--------+-----+-----+--------+
| id | name    | age | sex | salary | did | did | dname  |
+----+---------+-----+-----+--------+-----+-----+--------+
|  1 | alex    |  28 ||  53000 |   1 |   1 | python |
|  2 | wupeiqi |  23 ||  29000 |   1 |   1 | python |
|  3 | egon    |  30 ||  27000 |   1 |   1 | python |
|  4 | oldboy  |  22 ||      1 |   2 |   2 | linux  |
|  5 | jinxin  |  33 ||  28888 |   1 |   1 | python |
|  6 | 張無忌  |  20 ||   8000 |   3 |   3 | 明教   |
|  7 | 令狐沖  |  22 ||   6500 |   2 |   2 | linux  |
+----+---------+-----+-----+--------+-----+-----+--------+
7 rows in set
View Code

4,全連接查詢(顯示左右表中全部數據)

  全連接查詢:是在內連接的基礎上增加 左右兩邊沒有顯示的數據
  注意: mysql并不支持全連接 full JOIN 關鍵字
  注意: 但是mysql 提供了 UNION 關鍵字.使用 UNION 可以間接實現 full JOIN 功能

#查詢人員和部門的所有數據select * from person left join dept on person.did = dept.did
union 
select * from person right join dept on person.did = dept.did;

?

mysql> SELECT * FROM person LEFT JOIN dept ON person.did = dept.didUNION SELECT * FROM person RIGHT JOIN dept ON person.did = dept.did;
+------+----------+------+------+--------+------+------+--------+
| id   | name     | age  | sex  | salary | did  | did  | dname  |
+------+----------+------+------+--------+------+------+--------+
|    1 | alex     |   28 ||  53000 |    1 |    1 | python |
|    2 | wupeiqi  |   23 ||  29000 |    1 |    1 | python |
|    3 | egon     |   30 ||  27000 |    1 |    1 | python |
|    5 | jinxin   |   33 ||  28888 |    1 |    1 | python |
|    4 | oldboy   |   22 ||      1 |    2 |    2 | linux  |
|    7 | 令狐沖   |   22 ||   6500 |    2 |    2 | linux  |
|    6 | 張無忌   |   20 ||   8000 |    3 |    3 | 明教   |
|    8 | 東方不敗 |   23 ||  18000 | NULL | NULL | NULL   |
| NULL | NULL     | NULL | NULL | NULL   | NULL |    4 | 基督教 |
+------+----------+------+------+--------+------+------+--------+
9 rows in set注意: UNIONUNION ALL 的區別:UNION 會去掉重復的數據,而 UNION ALL 則直接顯示結果
View Code

三,復雜條件多表查詢

  1. 查詢出 教學部 年齡大于20歲,并且工資小于40000的員工,按工資倒序排列.(要求:分別使用多表聯合查詢和內連接查詢)

#1.多表聯合查詢方式:
select * from person p1,dept d2 where p1.did = d2.did  and d2.dname='python' and  age>20 and salary <40000 
ORDER BY salary DESC;#2.內連接查詢方式:
SELECT * FROM person p1 INNER JOIN dept d2 ON p1.did= d2.did and d2.dname='python' and  age>20 and salary <40000 
ORDER BY salary DESC;    
View Code

  2.查詢每個部門中最高工資和最低工資是多少,顯示部門名稱

select MAX(salary),MIN(salary),dept.dname from person LEFT JOIN deptON person.did = dept.didGROUP BY person.did;
View Code

四,子語句查詢

子查詢(嵌套查詢): 查多次, 多個select

注意: 第一次的查詢結果可以作為第二次的查詢的 條件 或者 表名 使用.

子查詢中可以包含:IN、NOT IN、ANY、ALL、EXISTS 和 NOT EXISTS等關鍵字.?還可以包含比較運算符:= 、 !=、> 、<等.

  1,作為表名使用

select * from (select * from person) as 表名;ps:大家需要注意的是: 一條語句中可以有多個這樣的子查詢,在執行時,
最里層括號(sql語句) 具有優先執行權.<br>注意: as 后面的表名稱不能加引號('')

  ?2,求最大工資那個人的姓名和薪水

1.求最大工資
select max(salary) from person;
2.求最大工資那個人叫什么
select name,salary from person where salary=53000;合并
select name,salary from person where salary=(select max(salary) from person);
View Code

  ?3,求工資高于所有人員平均工資的人員

1.求平均工資
select avg(salary) from person;2.工資大于平均工資的 人的姓名、工資
select name,salary from person where salary > 21298.625;合并
select name,salary from person where salary >(select avg(salary) from person);
View Code

  4.練習

    1.查詢平均年齡在20歲以上的部門名

    2.查詢教學部 下的員工信息

    3.查詢大于所有人平均工資的人員的姓名與年齡

#1.查詢平均年齡在20歲以上的部門名
SELECT * from dept where dept.did in (select dept_id from person GROUP BY dept_id HAVING avg(person.age) > 20
);#2.查詢教學部 下的員工信息
select * from person where dept_id = (select did from dept where dname ='教學部');#3.查詢大于所有人平均工資的人員的姓名與年齡
select * from person where salary > (select avg(salary) from person);
View Code

  ?5.關鍵字

假設any內部的查詢語句返回的結果個數是三個,如:result1,result2,result3,那么,select ...from ... where a > any(...);
->
select ...from ... where a > result1 or a > result2 or a > result3;
any 關鍵字

?

ALL關鍵字與any關鍵字類似,只不過上面的or改成and。即:select ...from ... where a > all(...);
->
select ...from ... where a > result1 and a > result2 and a > result3;
all 關鍵字
some關鍵字和any關鍵字是一樣的功能。所以:select ...from ... where a > some(...);
->
select ...from ... where a > result1 or a > result2 or a > result3;
some 關鍵字
EXISTSNOT EXISTS 子查詢語法如下:SELECT ... FROM table WHERE  EXISTS (subquery)
該語法可以理解為:主查詢(外部查詢)會根據子查詢驗證結果(TRUE 或 FALSE)來決定主查詢是否得以執行。mysql> SELECT * FROM person-> WHERE EXISTS-> (SELECT * FROM dept WHERE did=5);
Empty set (0.00 sec)
此處內層循環并沒有查詢到滿足條件的結果,因此返回false,外層查詢不執行。NOT EXISTS剛好與之相反mysql> SELECT * FROM person -> WHERE NOT EXISTS -> (SELECT * FROM dept WHERE did=5);
+----+----------+-----+-----+--------+------+
| id | name     | age | sex | salary | did  |
+----+----------+-----+-----+--------+------+
|  1 | alex     |  28 ||  53000 |    1 |
|  2 | wupeiqi  |  23 ||  29000 |    1 |
|  3 | egon     |  30 ||  27000 |    1 |
|  4 | oldboy   |  22 ||      1 |    2 |
|  5 | jinxin   |  33 ||  28888 |    1 |
|  6 | 張無忌   |  20 ||   8000 |    3 |
|  7 | 令狐沖   |  22 ||   6500 |    2 |
|  8 | 東方不敗 |  23 ||  18000 | NULL |
+----+----------+-----+-----+--------+------+
8 rows in set當然,EXISTS關鍵字可以與其他的查詢條件一起使用,條件表達式與EXISTS關鍵字之間用AND或者OR來連接,如下:mysql> SELECT * FROM person -> WHERE AGE >23 AND NOT EXISTS -> (SELECT * FROM dept WHERE did=5);
提示:
?EXISTS (subquery) 只返回 TRUE 或 FALSE,因此子查詢中的 SELECT * 也可以是 SELECT 1 或其他,官方說法是實際執行時會忽略 SELECT 清單,因此沒有區別。
exists 關鍵字

?五,其他查詢

  1,臨時表查詢

    需求:? 查詢高于本部門平均工資的人員

     解析思路: 1.先查詢本部門人員平均工資是多少.

    ? ?     ? 2.再使用人員的工資與部門的平均工資進行比較

#1.先查詢部門人員的平均工資
SELECT dept_id,AVG(salary)as sal from person GROUP BY dept_id;#2.再用人員的工資與部門的平均工資進行比較
SELECT * FROM person as p1,(SELECT dept_id,AVG(salary)as '平均工資' from person GROUP BY dept_id) as p2
where p1.dept_id = p2.dept_id AND p1.salary >p2.`平均工資`;ps:在當前語句中,我們可以把上一次的查詢結果當前做一張表來使用.因為p2表不是真是存在的,所以:我們稱之為 臨時表  臨時表:不局限于自身表,任何的查詢結果集都可以認為是一個臨時表.
View Code

2,判斷查詢? IF關鍵字

?   需求1 :根據工資高低,將人員劃分為兩個級別,分別為 高端人群和低端人群。顯示效果:姓名,年齡,性別,工資,級別

select p1.*, IF(p1.salary >10000,'高端人群','低端人群') as '級別'from person p1;#ps: 語法: IF(條件表達式,"結果為true",'結果為false');
View Code

    需求2: 根據工資高低,統計每個部門人員收入情況,劃分為 富人,小資,平民,吊絲 四個級別, 要求統計四個級別分別有多少人

#語法一:
SELECTCASE WHEN STATE = '1' THEN '成功'WHEN STATE = '2' THEN '失敗'ELSE '其他' END 
FROM 表;#語法二:
SELECT CASE age WHEN 23 THEN '23歲'WHEN 27 THEN '27歲'WHEN 30 THEN '30歲'ELSE '其他歲' END
FROM person;

  

SELECT dname '部門',sum(case WHEN salary >50000 THEN 1 ELSE 0 end) as '富人',sum(case WHEN salary between 29000 and 50000 THEN 1 ELSE 0 end) as '小資',sum(case WHEN salary between 10000 and 29000 THEN 1 ELSE 0 end) as '平民',sum(case WHEN salary <10000 THEN 1 ELSE 0 end) as '吊絲'
FROM person,dept where person.dept_id = dept.did GROUP BY dept_id
View Code

六,外鍵約束

1,問題

  什么是約束?

    約束是一種限制,它通過對表的行或列的數據做出限制,來確保表的數據的完整性、唯一性

  以上兩個表person和dept中,新人員可以沒有部門嗎? ? ?? 不可以

  新人員可以添加一個不存在的部門嗎? ?? 不可以

2,解決上面問題

?

  簡單的說,就是對兩個表的關系進行一些約束?(即: froegin key).?

  foreign key 定義:就是表與表之間的某種約定的關系,由于這種關系的存在,能夠讓表與表之間的數據,更加的完整,關連性更強。

3,具體操作(外鍵)

  3.1,創建表時,同時創建外鍵約束

CREATE TABLE IF NOT EXISTS dept (did int not null auto_increment PRIMARY KEY,dname VARCHAR(50) not null COMMENT '部門名稱'
)ENGINE=INNODB DEFAULT charset utf8;CREATE TABLE IF NOT EXISTS person(id int not null auto_increment PRIMARY KEY,name VARCHAR(50) not null,age TINYINT(4) null DEFAULT 0,sex enum('','','人妖') NOT NULL DEFAULT '人妖',salary decimal(10,2) NULL DEFAULT '250.00',hire_date date NOT NULL,dept_id int(11) DEFAULT NULL, CONSTRAINT fk_did FOREIGN KEY(dept_id) REFERENCES dept(did) -- 添加外鍵約束
)ENGINE = INNODB DEFAULT charset utf8;
View Code

  3.2 ,已經創建表后,追加外鍵約束

#添加外鍵約束
ALTER table person add constraint fk_did FOREIGN key(dept_id) REFERENCES dept(did);#刪除外鍵約束
ALTER TABLE person drop FOREIGN key fk_did;
View Code

4,定義外鍵的條件:

  (1)外鍵對應的字段數據類型保持一致,且被關聯的字段(即references指定的另外一個表的字段),必須保證唯一

  (2)所有tables的存儲引擎必須是InnoDB類型.

  (3)外鍵的約束4種類型: 1.RESTRICT 2.?NO ACTION 3.CASCADE 4.SET NULL

RESTRICT
同no action, 都是立即檢查外鍵約束NO ACTION
如果子表中有匹配的記錄,則不允許對父表對應候選鍵進行update/delete操作  CASCADE
在父表上update/delete記錄時,同步update/delete掉子表的匹配記錄 SET NULL
在父表上update/delete記錄時,將子表上匹配記錄的列設為null (要注意子表的外鍵列不能為not null)  
約束類型詳解

5,建議

  1,如果需要外鍵約束,最好創建表同時創建外鍵約束.

  2,如果需要設置級聯關系,刪除時最好設置為 SET NULL

  注:插入數據時,先插入主表中的數據,再插入從表中的數據。

? ? ?   ?刪除數據時,先刪除從表中的數據,再刪除主表中的數據。

?

七,其他約束類型

?1,非空約束

  關鍵字:not null 表示不可空。用來約束表中的字段列

create table t1(id int(10) not null primary key,name varchar(100) null);   

2,主鍵約束

  用于約束表中的一行,作為這一行的標識符,在一張表中通過主鍵就能準確定為到一行,因為此主鍵十分重要。

create table t2(id int(10) not null primary key
); 

  注意: 主鍵這一行的數據不能重復不能為空

3,復合主鍵

  主鍵不僅可以是表中的一列,也可以由表中的兩列或多列來共同標識

create table t3(id int(10) not null,name varchar(100) ,primary key(id,name)
);

4,唯一約束

  關鍵字:unique 它規定一張表中指定的一列的值必須不能有重復值,即這一列每一個值都是唯一的。

create table t4(id int(10) not null,name varchar(255) ,unique id_name(id,name)
);
//添加唯一約束
alter table t4 add unique id_name(id,name);
//刪除唯一約束
alter table t4 drop index id_name;

  注意: 當INSERT語句新插入的數據和已有數據重復的時候,如果有UNIQUE約束,則INSERT失敗.

5,默認值約束

  關鍵字:default

create table t5(id int(10) not null primary key,name varchar(255) default '張三'   
);
#插入數據
INSERT into t5(id) VALUES(1),(2);

  注意: INSERT語句執行時.,如果被DEFAULT約束的位置沒有值,那么這個位置將會被DEFAULT的值填充

八,表與表之間的關系

1,表關系分類:

  總體可以分為三類:一對一、一對多(多對一)、多對多

2,如何區分表與表之間是什么關系?

#分析步驟:
#多對一 /一對多
#1.站在左表的角度去看右表(情況一)
如果左表中的一條記錄,對應右表中多條記錄.那么他們的關系則為 一對多 關系.約束關系為:左表普通字段, 對應右表foreign key 字段.注意:如果左表與右表的情況反之.則關系為 多對一 關系.約束關系為:左表foreign key 字段, 對應右表普通字段.#一對一
#2.站在左表的角度去看右表(情況二)
如果左表中的一條記錄 對應 右表中的一條記錄. 則關系為 一對一關系.
約束關系為:左表foreign key字段上 添加唯一(unique)約束, 對應右表 關聯字段.
或者:右表foreign key字段上 添加唯一(unique)約束, 對應右表 關聯字段.#多對多
#3.站在左表和右表同時去看(情況三)
如果左表中的一條記錄 對應 右表中的多條記錄,并且右表中的一條記錄同時也對應左表的多條記錄. 那么這種關系 則 多對多 關系. 
這種關系需要定義一個這兩張表的[關系表]來專門存放二者的關系
View Code

3.建立表關系  

  3.1,一對多關系

  例如:一個人可以擁有多輛汽車,要求查詢某個人擁有的所有車輛。
  分析:人和車輛分別單獨建表,那么如何將兩個表關聯呢?有個巧妙的方法,在車輛的表中加個外鍵字段(人的編號)即可。
  * (思路小結:’建兩個表,一’方不動,’多’方添加一個外鍵字段)*

//建立人員表
CREATE TABLE people(id VARCHAR(12) PRIMARY KEY,sname VARCHAR(12),age INT,sex CHAR(1)
);
INSERT INTO people VALUES('H001','小王',27,'1');
INSERT INTO people VALUES('H002','小明',24,'1');
INSERT INTO people VALUES('H003','張慧',28,'0');
INSERT INTO people VALUES('H004','李小燕',35,'0');
INSERT INTO people VALUES('H005','王大拿',29,'1');
INSERT INTO people VALUES('H006','周強',36,'1');//建立車輛信息表
CREATE TABLE car(id VARCHAR(12) PRIMARY KEY,mark VARCHAR(24),price NUMERIC(6,2),pid VARCHAR(12),CONSTRAINT fk_people FOREIGN KEY(pid) REFERENCES people(id)
);
INSERT INTO car VALUES('C001','BMW',65.99,'H001');
INSERT INTO car VALUES('C002','BenZ',75.99,'H002');
INSERT INTO car VALUES('C003','Skoda',23.99,'H001');
INSERT INTO car VALUES('C004','Peugeot',20.99,'H003');
INSERT INTO car VALUES('C005','Porsche',295.99,'H004');
INSERT INTO car VALUES('C006','Honda',24.99,'H005');
INSERT INTO car VALUES('C007','Toyota',27.99,'H006');
INSERT INTO car VALUES('C008','Kia',18.99,'H002');
INSERT INTO car VALUES('C009','Bentley',309.99,'H005');
示例

3.2,一對一關系

?

?例如:一個中國公民只能有一個身份證信息

?  分析: 一對一的表關系實際上是 變異了的 一對多關系. 通過在從表的外鍵字段上添加唯一約束(unique)來實現一對一表關系.

 #身份證信息表
CREATE TABLE card (id int NOT NULL AUTO_INCREMENT PRIMARY KEY,code varchar(18) DEFAULT NULL,UNIQUE un_code (CODE) -- 創建唯一索引的目的,保證身份證號碼同樣不能出現重復
);INSERT INTO card VALUES(null,'210123123890890678'),(null,'210123456789012345'),(null,'210098765432112312');#公民表
CREATE TABLE people (id int NOT NULL AUTO_INCREMENT PRIMARY KEY,name varchar(50) DEFAULT NULL,sex char(1) DEFAULT '0',c_id int UNIQUE, -- 外鍵添加唯一約束,確保一對一CONSTRAINT fk_card_id FOREIGN KEY (c_id) REFERENCES card(id)
);INSERT INTO people VALUES(null,'zhangsan','1',1),(null,'lisi','0',2),(null,'wangwu','1',3);
示例

  3.3,多對多關系

  例如:學生選課,一個學生可以選修多門課程,每門課程可供多個學生選擇。
  分析:這種方式可以按照類似一對多方式建表,但冗余信息太多,好的方式是實體和關系分離并單獨建表,實體表為學生表和課程表,關系表為選修表,
  其中關系表采用聯合主鍵的方式(由學生表主鍵和課程表主鍵組成)建表。

#//建立學生表
CREATE TABLE student(id VARCHAR(10) PRIMARY KEY,sname VARCHAR(12),age INT,sex CHAR(1)
);
INSERT INTO student VALUES('S0001','王軍',20,1);
INSERT INTO student VALUES('S0002','張宇',21,1);
INSERT INTO student VALUES('S0003','劉飛',22,1);
INSERT INTO student VALUES('S0004','趙燕',18,0);
INSERT INTO student VALUES('S0005','曾婷',19,0);
INSERT INTO student VALUES('S0006','周慧',21,0);
INSERT INTO student VALUES('S0007','小紅',23,0);
INSERT INTO student VALUES('S0008','楊曉',18,0);
INSERT INTO student VALUES('S0009','李杰',20,1);
INSERT INTO student VALUES('S0010','張良',22,1);# //建立課程表
CREATE TABLE course(id VARCHAR(10) PRIMARY KEY,sname VARCHAR(12),credit DOUBLE(2,1),teacher VARCHAR(12)
);
INSERT INTO course VALUES('C001','Java',3.5,'李老師');
INSERT INTO course VALUES('C002','高等數學',5.0,'趙老師');
INSERT INTO course VALUES('C003','JavaScript',3.5,'王老師');
INSERT INTO course VALUES('C004','離散數學',3.5,'卜老師');
INSERT INTO course VALUES('C005','數據庫',3.5,'廖老師');
INSERT INTO course VALUES('C006','操作系統',3.5,'張老師');# //建立選修表
CREATE TABLE sc(sid VARCHAR(10),cid VARCHAR(10),PRIMARY KEY(sid,cid),CONSTRAINT fk_student FOREIGN KEY(sid) REFERENCES student(id),CONSTRAINT fk_course FOREIGN KEY(cid) REFERENCES course(id)
);INSERT INTO sc VALUES('S0001','C001');
INSERT INTO sc VALUES('S0001','C002');
INSERT INTO sc VALUES('S0001','C003');
INSERT INTO sc VALUES('S0002','C001');
INSERT INTO sc VALUES('S0002','C004');
INSERT INTO sc VALUES('S0003','C002');
INSERT INTO sc VALUES('S0003','C005');
INSERT INTO sc VALUES('S0004','C003');
INSERT INTO sc VALUES('S0005','C001');
INSERT INTO sc VALUES('S0006','C004');
INSERT INTO sc VALUES('S0007','C002');
INSERT INTO sc VALUES('S0008','C003');
INSERT INTO sc VALUES('S0009','C001');
INSERT INTO sc VALUES('S0009','C005');
示例

?

轉載于:https://www.cnblogs.com/eternity-twinkle/p/10858100.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/251780.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/251780.shtml
英文地址,請注明出處:http://en.pswp.cn/news/251780.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

sqlserver 創建對某個存儲過程執行情況的跟蹤

有時候需要抓取執行存儲過程時某個參數的值&#xff0c;有時候程序調用存儲過程執行后結果不太對&#xff0c;不確定是程序的問題還是存儲過程的問題&#xff0c;需要單獨執行存儲過程看結果 即可用下面的方法 -- --創建對某個存儲過程的執行情況的跟蹤 --注意修改路徑 和 obje…

5.7 彈性盒子

彈性盒子定義彈性盒子 display&#xff1a;flex定義子元素排列方式 flex-diection定義子元素換行方式 flxe-wrap定義子元素對齊方式橫向對齊 justify-content縱向對齊 align-items 媒體查詢 media screen and (max-width:最大寬度)and &#xff08;min-width&#xff1a;最小…

4.navicat11激活教程,親測可用哦!

原文地址&#xff1a;http://blog.csdn.net/sanbingyutuoniao123/article/details/52589678Navicat是一款數據庫管理工具, 用于簡化, 開發和管理MySQL, SQL Server, SQLite, Oracle 和 PostgreSQL 的數據庫&#xff1b;Navicat數據模型工具以圖形化方式創建關聯式數據庫&#x…

漢諾塔問題深度剖析(python實現)

當我們學習一門編程語言的時候&#xff0c;都會遇到遞歸函數這個問題。而學習遞歸的一個經典案例就是漢諾塔問題。通過這篇文章&#xff0c;觀察移動三個盤子和四個盤子的詳細過程&#xff0c;您不僅可以深刻的了解遞歸&#xff0c;也更加熟悉了漢諾塔的游戲的玩法。 更好的閱讀…

iOS-QQ臨時對話、QQ群申請跳轉

QQ 臨時對話 NSString *qq [NSString stringWithFormat:"mqq://im/chat?chat_typewpa&uin%&&version1&src_typeweb","這是是QQ號碼"];NSURL *urlQQ [NSURL URLWithString:qq];[[UIApplication sharedApplication] openURL:urlQQ]; QQ 申…

[luoguP2331] [SCOI2005]最大子矩陣(DP)

傳送門 orz不會做。。。 一個好理解的做法&#xff08;n^3*k&#xff09;&#xff1a; 分n1和n2兩種情況考慮。 n1時&#xff0c;預處理出前綴和sum[]。 設f[i][j]為到達第i格&#xff0c;已經放了j個子矩陣的最大和&#xff0c; 那么每次先把f[i][j]的值設為f[i-1][j]&#xf…

想要去阿里面試?你必須得跨過 JVM 這道坎!

概述 很多人想要到阿里巴巴、美團、京東等互聯網大公司去面試&#xff0c;但是現在互聯網大廠面試一般都必定會考核JVM相關的知識積累和實踐經驗&#xff0c;畢竟線上系統寫好代碼部署之后&#xff0c;每個工程師都必須關注JVM相關的東西&#xff0c;比如OOM、GC等問題. 所以一…

醫學知識圖譜一

大綱 知識自動提取技術 醫學知識融合 醫學知識推理 轉載于:https://www.cnblogs.com/quietwalk/p/9000950.html

在一個div里,列表樣式圖片進行float,實現水平排序

<div class"xiangce"><ul> <li><a href"#"><img src"images/pic4.gif" alt"">產品名稱</a></li><li><a href"#"><img src"images/pic4.gif" alt"…

團隊開發git使用各種問題

參考:https://www.cnblogs.com/schaepher/p/4933873.html 問題-3:保持github上項目干凈&#xff0c;對于在不同機器上運行會不同的文件不予維護(如.idea/workspace.xml) 建議:對于項目輸出在項目目錄中的文件不予維護 對于IDE自動生成且與項目所在目錄有關的文件不予維護 將這些…

filebeat 亂碼

查看 文件的類型 [rootelk-node-1 rsyslog] # file 192.168.1.16.log 192.168.1.16.log: Non-ISO extended-ASCII text, with very long lines, with LF, NEL line terminators 如果命令返回結果說明改日志為utf-8&#xff0c;則logstash配置文件中charset設置為UTF-8 如果命令…

團隊編程項目代碼設計規范(爬取豆瓣電影top250)

基本格式 縮進 使用4個空格進行縮進 行寬 每行代碼盡量不超過80個字符 理由&#xff1a; 這在查看side-by-side的diff時很有幫助方便在控制臺下查看代碼太長可能是設計有缺陷換行 Python支持括號內的換行。這時有兩種情況。 第二行縮進到括號的起始處foo long_function_name(v…

程序員的浪漫

程序員的浪漫 馬上就到520了&#xff0c;各位小伙伴想好了準備什么禮物送個自己的另一半呢&#xff1f;還沒想好的注意啦&#xff01;&#xff01;現在還有機會&#xff0c;今天給大家分享一些程序員的浪漫創意禮物&#xff0c;希望你可以從中找到一些靈感。 One Link&#xff…

14-1 部署項目

1313轉載于:https://www.cnblogs.com/ZHONGZHENHUA/p/9011671.html

The listener supports no services

$ lsnrctl start 報錯提示: The listener supports no services The command completed successfully 如圖所示&#xff1a; 這樣啟動后遠程連接會報錯&#xff1a; oracle ORA-12514:TNS:listener does not currently know of service requested in connect descriptor 問題原…

Luogu P2577 [ZJOI2005]午餐

一道貪心類背包DP的好題 首先發現一個十分顯然的性質&#xff0c;沒有這個性質整道題目都難以下手&#xff1a; 無論兩隊的順序如何&#xff0c;總是讓吃飯慢的人先排隊 這是一個很顯然的貪心&#xff0c;因為如果讓吃飯慢的排在后面要更多的時間至少沒有這樣優 因此我們先按吃…

SEO【總結】by 2019年5月

2019獨角獸企業重金招聘Python工程師標準>>> 關鍵點&#xff1a; 1、代碼 1.1、seo前端代碼&#xff1a;基于Html代碼的SEOherf&#xff1a;https://my.oschina.net/u/2862573/blog/3030664 注意的要點&#xff1a; h1&#xff0c;h2的內容很關鍵 網頁的壓縮、靜態化…

Linux 系統的啟動順序

第一步&#xff1a;加載BIOS當你打開ia計算機的電源&#xff0c;計算機會首先加載計算機主板的BIOS信息&#xff0c;因為它包含了CPU的相關信息&#xff0c;設備啟動順序[安裝系統的U盤啟動順序]&#xff0c;內存信息&#xff0c;時鐘信息&#xff0c;PnP特性等等&#xff0c; …

Oracle數據庫 查看表是否是 索引組織表的方法

1. 最近在工作過程中發現 一個表插入很慢 以為是索引組織表, 所以一直有點糾結 但是發現 產品里面是沒有IOT的 于是找了下公司的OCP 問了下 如何查看 就是 user_tables 視圖里面的一個字段. 見圖: 轉載于:https://www.cnblogs.com/jinanxiaolaohu/p/9018037.html

Windows server 2016 搭建RDS服務

計算機的更新換代太快&#xff0c;新購置的計算機沒幾年便覺得運行速度越來越慢&#xff0c;尤其是在運行一些比較大的應用程序是&#xff0c;用戶總是抱怨運行速度太慢或者總是死機等問題。如果要更換新的計算機&#xff0c;又得不到領導的批準&#xff0c;因此對于企業來說&a…