JavaWeb基础教程-----(1)MySQL

1. 数据库相关概念

  1. 数据库
    存储数据的仓库,数据是有组织的进行存储
    英文: DataBase, 简称DB

  2. 数据库管理系统
    管理数据库的大型软件
    英文: DataBase Management System,简称DBMS

  3. SQL
    英文: Structured Query Language,简称SQL,结构化查询语言
    操作关系型数据库的编程语言
    定义操作所有关系型数据库的统一标准

2. MySQL

2.1 MySQL安装

参考这篇MySQL安装配置教程

2.2 MySQL数据模型

关系型数据库:
关系型数据库是建立在关系模型基础上的数据库,简单说,关系型数据库是由多张能互相连接的二维表组成的数据库

优点:

  1. 都是使用表结构,格式一致,易于维护。
  2. 使用通用的 SQL语言操作,使用方便,可用于复杂查询。
  3. 数据存储在磁盘中, 安全。

3. Navicat(图形化客户端工具)

参考这篇MySQL+Navicat安装配置教程

4. SQL

4.1 SQL简介

  • 英文::Structured Query Language, 简称SQL
  • 结构化查询语言,一 门操作关系型数据库的编程语言
  • 定义操作所有关系型数据库的统一 标准
  • 对于同一个需求,每一种数据库操作的方式可能会存在一些不一样的地方,我们称为"方言”

4.2 SQL通用语法

  1. SQL语句可以单行或多行书写,以分号结尾。
  2. MySQL 数据库的SQL语句不区分大小写,关键字建议使用大写。
  3. 注释
    单行注释:-- 注释内容或#注释内容(MySQL特有)
    多行注释:/* 注释*/

注:单行注释-- 注释内容, --后的空格不能省略

4.3 SQL分类

  • DDL(Data Definition Language)数据定义语言,用来定义数据库对象:数据库,表,列等
  • DML(Data Manipulation Language)数据操作语言, 用来对数据库中表的数据进行增删改
  • DQL(Data Query Language)数据查询语言,用来查询数据库中表的记录(数据)
  • DCL(Data Control Language)数据控制语言,用来定义数据库的访问权限和安全级别,及创建用户

4.4 DDL

4.4.1 DDL----操作数据库

  1. 查询
SHOW DATABASES;
  1. 创建
CREATE DATABASE 数据库名称;
# CREATE DATABASE IF NOT EXISTS 数据库名称;
  1. 删除
DROP DATABASE 数据库名称;
#DROP DATABASE IF EXISTS 数据库名称;
  1. 使用数据库

查看当前使用的数据库:

SELECT DATABASE();

使用数据库:

USE 数据库名称;

4.4.2 MySQL自带数据库

information_schema: 是信息数据库,其中保存着关于MySQL服务器所维护的所有其他数据库的信息。如数据库名,数据库的表,表栏的数据类型与访问权 限等。在INFORMATION_SCHEMA中,有数个只读表。它们实际上是视图,而不是基本表,因此,无法看到与之相关的任何文件。

mysql: 核心数据库,存储MySQL数据库里最核心的信息,例如权限、安全。

sys: 系统数据库。

performance_schema: 主要用于收集数据库服务器性能参数(研究性能调优要用到)

4.4.3 DDL----操作表

创建(Create) 查询(Retrieve) 修改(Update) 删除 (Delete)

查询表:

  • 查询当前数据库下的所有表名称
SHOW TABLES;
  • 查询表结构
DESC 表名称;

创建表:

CREATE TABLE 表名 (
	字段名1 数据类型1,
	字段名2 数据类型2,
	...
	字段名n 数据类型n
);

数据类型:
MySQL支持多种类型,可以分为三类:数值、日期、字符串
在这里插入图片描述

举例:
int类型: age int
double类型: 字段名 double(总长度,小数点后保留的位数)【score double(5,2)】
data类型: birthday date (注意:日期值要带引号,而且只包含年月日)
char类型: 定长字符串。优点:存储性能高;缺点:浪费空间
name char(10) 如果存储的数据字符个数不足10个,也会占10个的空间,汉字占1个字符
varchar类型: 变长字符串。优点:节约空间;缺点:存储性能底
name varchar(10) 如果存储的数据字符个数不足10个,那就数据字符个数是几就占几个的空间

删除表:

DROP TABLE 表名;
#DROP TABLE IF EXISTS 表名;

修改表:

  1. 修改表名
ALTER TABLE 表名 RENAME TO 新的表名;
  1. 添加一列
ALTER TABLE 表名 ADD 列名 数据类型;
  1. 修改数据类型
ALTER TABLE 表名 MODIFY 列名 新数据类型;
  1. 修改列名和数据类型
ALTER TABLE 表名 CHANGE 列名 新列名 新数据类型;
  1. 删除列
ALTER TABLE 表名 DROP 列名;
  1. 删除外键约束
ALTER TABLE 表名 DROP FOREIGN KEY 外键名称;

4.5 DML

4.5.1 DML----添加数据

  1. 给指定列添加数据
INSERT INTO 表名(列名1,列名2,) VALUES(1,2,);
  1. 给全部列添加数据
INSERT INTO 表名 VALUES(1,2,);
  1. 批量添加数据
 INSERT INTO 表名(列名1,列名2,) 
 VALUES(1,2,),(1,2,),(1,2,);
INSERT INTO 表名 VALUES(1,2,),(1,2,),(1,2,);

【示例】DML----添加数据

-- 查询所有数据
SELECT
	* 
FROM
	student;
	
-- 给指定列添加数据
INSERT INTO student ( id, NAME )
VALUES
	( 1, '张三' );
	
-- 给所有列添加数据,列名的列表可以省略的(但是不建议)
INSERT INTO student ( id, NAME, gender, birthday, score, email, tel, STATUS )
VALUES
	( 2, '李四', '男', '1999-11-11', 88.88, 'lisi@itcast.cn', '13888888888', 1 );
INSERT INTO student
VALUES
	( 2, '李四', '男', '1999-11-11', 88.88, 'lisi@itcast.cn', '13888888888', 1 );
INSERT INTO student ( id, NAME, gender, birthday, score, email, tel, STATUS )
VALUES
	( 3, '李四', '男', '1999-11-11', 88.88, 'lisi@itcast.cn', '13888888888', 1 ),
	( 4, '李四', '男', '1999-11-11', 88.88, 'lisi@itcast.cn', '13888888888', 1 ),
	( 5, '李四', '男', '1999-11-11', 88.88, 'lisi@itcast.cn', '13888888888', 1 );

4.5.2 DML----修改数据

-- 修改数据:UPDATE 表名 SET 列名1=值1,列名2=值2,… [WHERE 条件];
UPDATE student SET gender = '女' WHERE name = '张三';

注意:修改语句中如果不加条件,则会将所有记录都修改!!!

4.5.3 DML----删除数据

-- 删除数据:DELETE FROM 表名 [WHERE 条件];
DELETE FROM student WHERE name = '张三';

注意:删除语句中如果不加条件,则会将所有记录都删除!!!

4.6 DQL

语法:

SELECT 
    字段列表
FROM 
    表名列表 
WHERE 
    条件列表
GROUP BY
    分组字段
HAVING
    分组后条件
ORDER BY
    排序字段
LIMIT
    分页限定

4.6.1 DQL----基础查询

  1. 查询多个字段
SELECT 字段列表1,字段列表2 FROM 表名;
SELECT * FROM 表名; -- 查询所有数据,*代替了所有的列名,但是不建议使用
  1. 去除重复记录
SELECT DISTINCT 字段列表 FROM 表名;
  1. 起别名
-- AS: AS 也可以省略
SELECT DISTINCT name AS '名字',age '年龄' FROM student;

4.6.2 DQL----条件查询(WHERE)

  1. 条件查询语法:
SELECT 字段列表 FROM 表名 WHERE 条件列表;
  1. 条件:
    在这里插入图片描述

【示例】条件查询

SELECT * FROM student where age >= 20;
SELECT * FROM student where age >= 20 && age <= 30;
SELECT * FROM student where age BETWEEN 20 AND 30;
-- 日期也可以使用>=
SELECT * FROM student where date BETWEEN '2020-01-01' AND '2021-01-01';
-- 等于
SELECT * FROM student where age = 18;
-- 不等于
SELECT * FROM student where age != 18;
SELECT * FROM student where age <> 18;
-- or
SELECT * FROM student where age = 18 OR age = 20;
SELECT * FROM student where age IN(18,20);

-- is null
SELECT * FROM student where english IS NULL;

注意:null值的比较不能使用= !=,要 IS NULL或 IS NOT NULL

  1. 模糊查询

在这里插入图片描述

【示例】模糊查询

-- 查询姓马的学生信息
SELECT * FROM student where name like '马%';
-- 查询第二个字是花的学生信息
SELECT * FROM student where name like '_花%';
-- 查询名字中包含德的学生信息
SELECT * FROM student where name like '%德%';

4.6.3 DQL----排序查询(GROUP BY)

  1. 排序查询语法
SELECT 字段列表 FROM 表名 
ORDER BY 排序字段名1 [排序方式1],排序字段名2 [排序方式2];
  1. 排序方式
    ASC : 升序排列(默认值)
    DESC : 降序排列

注意:如果有多个排序条件,当前边的条件值一样时,才会根据第二条件进行排序

4.6.4 DQL----聚合函数

  1. 概念: 将一列数据作为一个整体进行纵向计算
  2. 分类:
    函数名功能
    count(列名)统计数量(一般选用不为null的列)
    max(列名)最大值
    min(列名)最小值
    sum(列名)求和
    avg(列名)平均值
  3. 语法:
SELECT 聚合函数名(列名) FROM;

注意:null 值不参与所有聚合函数运算

【示例】聚合函数

-- count 一般使用count(主键)或者count(*)
select count(id) from student; -- 一般count统计的列名不能为空
select count(*) from student;
-- max
select max(math) from student;
-- min
select min(math) from student;
-- sum
select sum(math) from student;
-- avg
select avg(math) from student;

4.6.5 DQL----分组查询(ORDER BY)

  1. 分组查询语法:
SELECT 字段列表 FROM 表名 [WHERE 分组前条件限定] GROUP BY 分组字段名 [HAVING 分组后条件过滤];

注意:分组之后,查询的字段为聚合函数和分组字段,查询其他字段无任何意义。

  1. where 和 having 区别:

执行时机不一样:where 是分组之前进行限定,不满足where条件,则不参与分组,而having是分组之后对结果进行过滤。
可判断的条件不一样:where 不能对聚合函数进行判断,having 可以。

  1. 执行顺序:where>聚合函数>having,不可能判断后面执行的条件。

【示例】分组排序

-- 查询男同学和女同学各自的数学平均分
select sex, avg(math) from student group by sex;
-- 查询男同学和女同学各自的数学平均分,以及各自人数
select sex, avg(math),count(*) from student group by sex;
-- 查询男同学和女同学各自的数学平均分,以及各自人数,要求:分数低于70分的不参与分组
select sex, avg(math),count(*) from stu where math > 70 group by sex;
-- 查询男同学和女同学各自的数学平均分,以及各自人数,要求:分数低于70分的不参与分组,分组之后人数大于2个的
select sex, avg(math),count(*) from stu where math > 70 group by sex having count(*) > 2;

4.6.6 DQL----分页查询(LIMIT)

  1. 分页查询语法:
SELECT 字段列表 FROM 表名 LIMIT  起始索引 , 查询条目数; --起始索引从0开始
  1. 计算公式:起始索引 = (当前页码 - 1) * 每页显示的条数

【示例】分页查询

-- 从0开始查询,查询3条数据
select * from student limit 0 , 3;
-- 每页显示3条数据,查询第1页数据
select * from student limit 0 , 3;
-- 每页显示3条数据,查询第2页数据
select * from student limit 3 , 3;
-- 每页显示3条数据,查询第3页数据
select * from student limit 6 , 3;

注意:
分页查询limit是MySQL数据库的方言;
Oracle分页查询使用rownumber;
SQL Server分页查询使用top。

5. 约束

5.1 概念&分类

  1. 约束的概念:

    • 约束是作用于表中列上的规则,用于限制加入表的数据
    • 约束的存在保证了数据库中数据的正确性、有效性和完整性
  2. 约束的分类
    在这里插入图片描述

注意:MySQL不支持检查约束。

【示例】约束

DROP TABLE IF EXISTS emp;

-- 员工表
CREATE TABLE emp (
	id INT PRIMARY KEY auto_increment,	-- 员工id,主键且自增长
	ename VARCHAR(50) NOT NULL UNIQUE, -- 员工姓名,非空且唯一
	joindate DATE NOT NULL,	-- 入职日期非空
	salary DOUBLE(7,2) NOT NULL, -- 工资,非空
	bonus DOUBLE(7,2)	DEFAULT 0 -- 奖金,默认为0
);

5.2 非空约束

  1. 概念

非空约束用于保证列中所有的数据不为NULL值

  1. 语法
-- 创建表时添加非空约束
CREATE TABLE 表名 (
	列名 数据类型 NOT NULL,
	...
);
-- 建完表后添加非空约束
ALTER TABLE 表名 MODIFY 字段名 数据类型 NOT NULL;
-- 删除约束
ALTER TABLE 表名 MODIFY 字段名 数据类型;

5.3 唯一约束

  1. 概念
    唯一约束用于保证列中所有数据各不相同
  2. 语法
    在这里插入图片描述

5.4 主键约束

  1. 概念
    主键是一行数据的唯一标识,要求非空且唯一;
    一张表只有一个主键

  2. 语法
    在这里插入图片描述

5.5 默认约束

  1. 概念
    保存数据时,未指定值则采用默认值
  2. 语法
    在这里插入图片描述

5.6 外键约束

  1. 概念
    外键用来让两个表的数据之间建立链接,保证数据的一致性和完整性
  2. 语法
    在这里插入图片描述
    【示例】外键约束
-- 删除表
DROP TABLE IF EXISTS emp;
DROP TABLE IF EXISTS dept;
 
-- 部门表
CREATE TABLE dept(
	id int primary key auto_increment,
	dep_name varchar(20),
	addr varchar(20)
);

-- 员工表 
CREATE TABLE emp(
	id int primary key auto_increment,
	name varchar(20),
	age int,
	dep_id int,
 
	-- 添加外键 dep_id,关联 dept 表的id主键
	CONSTRAINT fk_emp_dept FOREIGN KEY(dep_id) REFERENCES dept(id)	
);

-- 添加 2 个部门
insert into dept(dep_name,addr) values
('研发部','广州'),('销售部', '深圳');-- 添加员工,dep_id 表示员工所在的部门
INSERT INTO emp (NAME, age, dep_id) VALUES 
('张三', 20, 1),
('李四', 20, 1),
('王五', 20, 1),
('赵六', 20, 2),
('孙七', 22, 2),
('周八', 18, 2);

-- 删除外键
alter table emp drop FOREIGN key fk_emp_dept;

-- 建完表后,添加外键
alter table emp add CONSTRAINT fk_emp_dept FOREIGN key(dep_id) REFERENCES dept(id);

注意:要先创建和添加都是先主表(部门表),再从表(员工表);删除先从表后主表。

6. 数据库设计

6.1 数据库设计简介

  1. 软件研发的步骤
    在这里插入图片描述
    产品原型参考网站:产品大牛

  2. 数据库设计的概念
    (1)数据库设计就是根据业务系统的具体需求,结合我们所选用的DBMS,为这个业务系统构造出最优的数据存储模型。
    (2)建立数据库中的表结构以及表与表之间的关联关系的过程。
    (3)有哪些表?表里有哪些字段?表和表之间有什么关系?

  3. 数据库设计的步骤
    (1)需求分析(数据是什么? 数据具有哪些属性? 数据与属性的特点是什么)
    (2)逻辑分析(通过ER图对数据库进行逻辑建模,不需要考虑我们所选用的数据库管理系统)如:在这里插入图片描述

(3)物理设计(根据数据库自身的特点把逻辑设计转换为物理设计)
(4)维护设计(1.对新的需求进行建表;2.表优化)

6.2 表关系

6.2.1 一对一

如:用户 和 用户详情
一对一关系多用于表拆分,将一个实体中经常使用的字段放一张表,不经常使用的字段放另一张表,用于提升查询性能。
实现方式: 在任意一方加入外键,关联另一方主键,并且设置外键为唯一(UNIQUE)

【示例】一对一(用户 和 用户详情)
在这里插入图片描述

create table tb_user_desc (
    id int primary key auto_increment,
    city varchar(20),
    edu varchar(10),
    income int,
    status char(2),
    des varchar(100)
);create table tb_user (
    id int primary key auto_increment,
    photo varchar(100),
    nickname varchar(50),
    age int,
    gender char(1),
    desc_id int unique,
    -- 添加外键
    CONSTRAINT fk_user_desc FOREIGN KEY(desc_id) REFERENCES tb_user_desc(id)    
);

在这里插入图片描述

6.2.2 一对多(多对一)

如:部门 和 员工;一个部门对应多个员工,一个员工对应一个部门。
实现方式: 在任意一方加入外键,关联另一方主键,并且设置外键为唯一(UNIQUE)
【示例】一对多(部门 和 员工)
在这里插入图片描述

-- 删除表
DROP TABLE IF EXISTS emp;
DROP TABLE IF EXISTS dept;
 
-- 部门表
CREATE TABLE dept(
	id int primary key auto_increment,
	dep_name varchar(20),
	addr varchar(20)
);

-- 员工表 
CREATE TABLE emp(
	id int primary key auto_increment,
	name varchar(20),
	age int,
	dep_id int,
 
	-- 添加外键 dep_id,关联 dept 表的id主键
	CONSTRAINT fk_emp_dept FOREIGN KEY(dep_id) REFERENCES dept(id)	
);

-- 添加 2 个部门
insert into dept(dep_name,addr) values
('研发部','广州'),('销售部', '深圳');-- 添加员工,dep_id 表示员工所在的部门
INSERT INTO emp (NAME, age, dep_id) VALUES 
('张三', 20, 1),
('李四', 20, 1),
('王五', 20, 1),
('赵六', 20, 2),
('孙七', 22, 2),
('周八', 18, 2);

在这里插入图片描述

6.2.3 多对多

如:商品 和 订单;一个商品对应多个订单,一个订单包含多个商品。
实现方式: 建立第三张中间表,中间表至少包含两个外键,分别关联两方主键。
在这里插入图片描述【示例】多对多(商品 和 订单)

-- 删除表
DROP TABLE IF EXISTS tb_order_goods;
DROP TABLE IF EXISTS tb_order;
DROP TABLE IF EXISTS tb_goods;
 
-- 订单表
CREATE TABLE tb_order(
	id int primary key auto_increment,
	payment double(10,2),
	payment_type TINYINT,
	status TINYINT
);
 
-- 商品表
CREATE TABLE tb_goods(
	id int primary key auto_increment,
	title varchar(100),
	price double(10,2)
);
 
-- 订单商品中间表
CREATE TABLE tb_order_goods(
	id int primary key auto_increment,
	order_id int,
	goods_id int,
	count int	-- 业务字段
);
 
-- 建完表后,添加外键
alter table tb_order_goods add CONSTRAINT fk_order_id FOREIGN key(order_id) REFERENCES tb_order(id);
alter table tb_order_goods add CONSTRAINT fk_goods_id FOREIGN key(goods_id) REFERENCES tb_goods(id);

在这里插入图片描述

6.2.4 数据库设计案例

7. 多表查询

7.1 多表查询简介

  • 笛卡尔积:取A,B集合所有组合情况

  • 多表查询:从多张表查询数据
    (1)连接查询

    • 内连接:相当于查询A B交集数据
    • 外连接:
      • 左外连接:相当于查询A表所有数据和交集部分数据
      • 右外连接:相当于查询B表所有数据和交集部分数据

    (2)子查询

7.2 内连接

  1. 语法
-- 隐式内连接。没有JOIN关键字,条件使用WHERE指定。书写简单,多表时效率低
SELECT 字段列表 FROM1,2WHERE 条件;
 
-- 显示内连接。使用INNER JOIN ... ON语句, 可以省略INNER。书写复杂,多表时效率高
SELECT 字段列表 FROM1 [INNER] JOIN2 ON 条件;
  1. 内连接相当于查询AB交集数据。
    在这里插入图片描述

【示例】创建部门和员工表

DROP TABLE IF EXISTS emp;
DROP TABLE IF EXISTS dept;

# 创建部门表
	CREATE TABLE dept(
        did INT PRIMARY KEY AUTO_INCREMENT,
        dname VARCHAR(20)
    );
 
	# 创建员工表
	CREATE TABLE emp (
        id INT PRIMARY KEY AUTO_INCREMENT,
        NAME VARCHAR(10),
        gender CHAR(1), -- 性别
        salary DOUBLE, -- 工资
        join_date DATE, -- 入职日期
        dep_id INT,
        FOREIGN KEY (dep_id) REFERENCES dept(did) -- 外键,关联部门表(部门表的主键)
    );
    
	-- 添加部门数据
	INSERT INTO dept (dNAME) VALUES ('研发部'),('市场部'),('财务部'),('销售部');
	
	-- 添加员工数据
	INSERT INTO emp(NAME,gender,salary,join_date,dep_id) VALUES
	('孙悟空','男',7200,'2013-02-24',1),
	('猪八戒','男',3600,'2010-12-02',2),
	('唐僧','男',9000,'2008-08-08',2),
	('白骨精','女',5000,'2015-10-07',3),
	('蜘蛛精','女',4500,'2011-03-14',1),
	('小白龙','男',2500,'2011-02-14',null);	

【示例】内连接

-- 隐式内连接
SELECT
	emp.name,
	emp.gender,
	dept.dname
FROM
	emp,
	dept
WHERE
	emp.dep_id = dept.did;

-- 显示内连接
select * from emp inner join dept on emp.dep_id = dept.did;

7.3 外连接

  1. 语法
-- 左外连接
SELECT 字段列表 FROM1 LEFT [OUTER] JOIN2 ON 条件;

-- 右外连接
SELECT 字段列表 FROM1 RIGHT [OUTER] JOIN2 ON 条件;
  1. 左外连接:相当于查询A表所有数据和交集部分数据
  2. 右外连接:相当于查询B表所有数据和交集部分数据

【示例】外连接查询

-- 左外连接
select * from emp left join dept on emp.dep_id = dept.did;

-- 右外连接
select * from emp right join dept on emp.dep_id = dept.did;

7.4 子连接

  1. 概念
    查询中嵌套查询,称嵌套查询为子查询。
  2. 子查询根据查询结果的不同、作用不同分为:单行单列、多行单列、多行多列。

7.4.1 单行单列

子查询语句作为条件值,使用 = != > < 等进行条件判断

  1. 语法:
SELECT 字段列表 FROMWHERE 字段名 = (子查询);
  1. 【示例】查询比猪八戒薪水高的员工:
SELECT * FROM emp WHERE salary >(SELECT salary FROM emp WHERE name='猪八戒');

7.4.2 多行单列

子查询语句作为条件值,使用 in 等关键字进行条件判断

  1. 语法:
SELECT 字段列表 FROMWHERE 字段名 in (子查询);
  1. 【示例】查询 ‘财务部’ 和 ‘市场部’ 所有的员工信息
SELECT * FROM emp WHERE dep_id in (SELECT did FROM dept WHERE dname IN ('财务部','市场部'));

7.4.3 多行多列

子查询语句作为虚拟表

  1. 语法:
SELECT 字段列表 FROM (子查询) WHERE 条件;
  1. 【示例】查询入职日期是 ‘2011-11-11’ 之后的员工信息和部门信息
select * from (select * from emp where join_date > '2011-11-11' ) t1, dept where t1.dep_id = dept.did;

7.5 多表查询案例

【示例】创建部门、员工、职务以及工资等级表

DROP TABLE IF EXISTS emp;
DROP TABLE IF EXISTS dept;
DROP TABLE IF EXISTS job;
DROP TABLE IF EXISTS salarygrade;
 
-- 部门表
CREATE TABLE dept (
  did INT PRIMARY KEY PRIMARY KEY, -- 部门id
  dname VARCHAR(50), -- 部门名称
  loc VARCHAR(50) -- 部门所在地
);
 
-- 职务表,职务名称,职务描述
CREATE TABLE job (
  id INT PRIMARY KEY,
  jname VARCHAR(20),
  description VARCHAR(50)
);
 
-- 员工表
CREATE TABLE emp (
  id INT PRIMARY KEY, -- 员工id
  ename VARCHAR(50), -- 员工姓名
  job_id INT, -- 职务id
  mgr INT , -- 上级领导
  joindate DATE, -- 入职日期
  salary DECIMAL(7,2), -- 工资
  bonus DECIMAL(7,2), -- 奖金
  dept_id INT, -- 所在部门编号
  CONSTRAINT emp_jobid_ref_job_id_fk FOREIGN KEY(job_id) REFERENCES job(id),
  CONSTRAINT emp_deptid_ref_dept_id_fk FOREIGN KEY(dept_id) REFERENCES dept(did)
);
-- 工资等级表
CREATE TABLE salarygrade (
  grade INT PRIMARY KEY,   -- 级别
  losalary INT,  -- 最低工资
  hisalary INT -- 最高工资
);
				
-- 添加4个部门
INSERT INTO dept(did,dname,loc) VALUES 
(10,'教研部','北京'),
(20,'学工部','上海'),
(30,'销售部','广州'),
(40,'财务部','深圳');
 
-- 添加4个职务
INSERT INTO job (id, jname, description) VALUES
(1, '董事长', '管理整个公司,接单'),
(2, '经理', '管理部门员工'),
(3, '销售员', '向客人推销产品'),
(4, '文员', '使用办公软件');
 
 
-- 添加员工
INSERT INTO emp(id,ename,job_id,mgr,joindate,salary,bonus,dept_id) VALUES 
(1001,'孙悟空',4,1004,'2000-12-17','8000.00',NULL,20),
(1002,'卢俊义',3,1006,'2001-02-20','16000.00','3000.00',30),
(1003,'林冲',3,1006,'2001-02-22','12500.00','5000.00',30),
(1004,'唐僧',2,1009,'2001-04-02','29750.00',NULL,20),
(1005,'李逵',4,1006,'2001-09-28','12500.00','14000.00',30),
(1006,'宋江',2,1009,'2001-05-01','28500.00',NULL,30),
(1007,'刘备',2,1009,'2001-09-01','24500.00',NULL,10),
(1008,'猪八戒',4,1004,'2007-04-19','30000.00',NULL,20),
(1009,'罗贯中',1,NULL,'2001-11-17','50000.00',NULL,10),
(1010,'吴用',3,1006,'2001-09-08','15000.00','0.00',30),
(1011,'沙僧',4,1004,'2007-05-23','11000.00',NULL,20),
(1012,'李逵',4,1006,'2001-12-03','9500.00',NULL,30),
(1013,'小白龙',4,1004,'2001-12-03','30000.00',NULL,20),
(1014,'关羽',4,1007,'2002-01-23','13000.00',NULL,10);
 
 
-- 添加5个工资等级
INSERT INTO salarygrade(grade,losalary,hisalary) VALUES 
(1,7000,12000),
(2,12010,14000),
(3,14010,20000),
(4,20010,30000),
(5,30010,99990);
  1. 查询所有员工信息。查询员工编号,员工姓名,工资,职务名称,职务描述
/*
    分析:
        1. 员工编号,员工姓名,工资 信息在emp 员工表中
        2. 职务名称,职务描述 信息在 job 职务表中
        3. job 职务表 和 emp 员工表 是 一对多的关系 emp.job_id = job.id
*/
-- 隐式内连接
SELECT
    emp.id,
    emp.ename,
    emp.salary,
    job.jname,
    job.description
FROM
    emp,
    job
WHERE
    emp.job_id = job.id;-- 显式内连接
SELECT
    emp.id,
    emp.ename,
    emp.salary,
    job.jname,
    job.description
FROM
    emp
INNER JOIN job ON emp.job_id = job.id;
  1. 查询员工编号,员工姓名,工资,职务名称,职务描述,部门名称,部门位置
/*
    分析:
        1. 员工编号,员工姓名,工资 信息在emp 员工表中
        2. 职务名称,职务描述 信息在 job 职务表中
        3. job 职务表 和 emp 员工表 是 一对多的关系 emp.job_id = job.id

        4. 部门名称,部门位置 来自于 部门表 dept
        5. dept 和 emp 一对多关系 dept.id = emp.dept_id
*/-- 隐式内连接
SELECT
    emp.id,
    emp.ename,
    emp.salary,
    job.jname,
    job.description,
    dept.dname,
    dept.loc
FROM
    emp,
    job,
    dept
WHERE
    emp.job_id = job.id
    and dept.id = emp.dept_id
;-- 显式内连接
SELECT
    emp.id,
    emp.ename,
    emp.salary,
    job.jname,
    job.description,
    dept.dname,
    dept.loc
FROM
    emp
INNER JOIN job ON emp.job_id = job.id
INNER JOIN dept ON dept.id = emp.dept_id
  1. 查询员工姓名,工资,工资等级
/*
    分析:
        1. 员工姓名,工资 信息在emp 员工表中
        2. 工资等级 信息在 salarygrade 工资等级表中
        3. emp.salary >= salarygrade.losalary  and emp.salary <= salarygrade.hisalary
*/
SELECT
    emp.ename,
    emp.salary,
    t2.*
FROM
    emp,
    salarygrade t2
WHERE
    emp.salary >= t2.losalary
AND emp.salary <= t2.hisalary
  1. 查询员工姓名,工资,职务名称,职务描述,部门名称,部门位置,工资等级
/*
    分析:
        1. 员工编号,员工姓名,工资 信息在emp 员工表中
        2. 职务名称,职务描述 信息在 job 职务表中
        3. job 职务表 和 emp 员工表 是 一对多的关系 emp.job_id = job.id
​
        4. 部门名称,部门位置 来自于 部门表 dept
        5. dept 和 emp 一对多关系 dept.id = emp.dept_id
        6. 工资等级 信息在 salarygrade 工资等级表中
        7. emp.salary >= salarygrade.losalary  and emp.salary <= salarygrade.hisalary
*/
SELECT
    emp.id,
    emp.ename,
    emp.salary,
    job.jname,
    job.description,
    dept.dname,
    dept.loc,
    t2.grade
FROM
    emp
INNER JOIN job ON emp.job_id = job.id
INNER JOIN dept ON dept.id = emp.dept_id
INNER JOIN salarygrade t2 ON emp.salary BETWEEN t2.losalary and t2.hisalary;
  1. 查询出部门编号、部门名称、部门位置、部门人数
/*
    分析:
        1. 部门编号、部门名称、部门位置 信息在dept 部门表中
        2. 部门人数:在emp表中 按照dept_id 进行分则,然后用count(*)来统计数量
        3. 使用子查询,让部门表和分组后的表进行内连接
*/

SELECT dept.id,dept.dname,dept.loc,t1.count FROM dept,(SELECT dept_id, count(*) count FROM emp GROUP BY dept_id) t1 WHERE dept.id = t1.dept_id;

8. 事务

8.1 事务简介

  • 数据库的事务(Transaction)是一种机制、一个操作序列,包含了一组数据库操作命令。
  • 事务把所有的命令作为一个整体一起向系统提交或撤销操作请求,即这一组数据库命令要么同时成功,要么同时失败。
  • 事务是一个不可分割的工作逻辑单元。
  • 例如: 在转账前开启事务,如果出现了异常回滚事务,三步正常执行就提交事务
    在这里插入图片描述
  • 语法:
-- 开启事务
START TRANSACTION
或者 BEGIN;
-- 提交事务
COMMIT;
-- 回滚事务
ROLLBACK;

8.2 事务操作

【示例】创建账户表

DROP TABLE IF EXISTS account;

-- 创建账户表
CREATE TABLE account(
	id int primary key auto_increment,
	name varchar(10),
	money double(10,2)
);

-- 添加数据
INSERT INTO acccount(name,money) values ('张三',1000),('李四',1000);

【示例】事务操作

-- 开启事务
BEGIN;
-- 转账操作
-- 1. 查询李四账户金额是否大于500
 
-- 2. 李四账户 -500
UPDATE account set money = money - 500 where name = '李四';
 
出错了...  -- 此处不是注释,在整体执行时会出问题,后面的sql则不执行
-- 3. 张三账户 +500
UPDATE account set money = money + 500 where name = '张三';
 
-- 提交事务
COMMIT;
 
-- 回滚事务
ROLLBACK;

8.3 事务四大特征

  • 原子性(Atomicity): 事务是不可分割的最小操作单位,要么同时成功,要么同时失败
  • 一致性(Consistency):事务完成时,必须使所有的数据都保持一致状态
  • 隔离性(Isolation):多个事务之间,操作的可见性
  • 持久性(Durability):事务一旦提交或回滚,它对数据库中的数据的改变就是永久的

注意:mysql中事务默认是自动提交的,Orcal事务默认是手动提交的。

-- 查看事务的默认提交方式
SELECT @@autocommit;
-- 1 自动提交 0 手动提交
-- 修改事务提交方式
set @@autocommit = 0;

参考文章

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值