JavaEE Day04 MySQL多表&事务

2023-02-16,,,,

今日内容

多表查询
事务
DCL用于控制权限和管理用户,DBA完成:SQL中四类DDL  DML  DQL  DCL

一、多表查询
1.多表查询_概述
1.1 查询语法
    select 
            列名列表
    from
            表名列表
    where
            ……
笛卡尔积:
        有两个集合A,B,取这两个集合的所有组成情况
        完成多表查询,需要消除无用的数据
1.2 多表查询的分类
内连接查询、外连接查询、子查询
2. 内连接查询

隐式内连接:使用where条件消除无用的数据

-- 隐式内连接
SELECT * FROM emp,dept WHERE emp.`dept_id`=dept.`id`;
-- 通常会查询某些字段
-- 查询员工表名称、性别及部门表的名称
SELECT emp.name,gender,dept.name FROM emp,dept WHERE emp.`dept_id`=dept.`id`;
-- 为表起别名,公司的实际写法,写到多行分开写
SELECT
t1.`name`,t1.`gender`,t2.`name`
FROM
emp t1,dept t2
WHERE
t1.`dept_id`=t2.`id`;
-- 更正规的写法
SELECT
t1.`name`,-- 员工表名称
t1.`gender',-- 员工表性别
t2.`NAME` -- 部门表名称
FROM
emp t1,
dept t2
WHERE
t1.`dept_id`=t2.`id`;

显式内连接
语法:select 字段列表 from表名1 [inner] join 表名2 on 条件

SELECT * FROM emp INNER JOIN dept ON emp.dept_id=dept.id;
SELECT * FROM emp JOIN dept ON emp.dept_id=dept.id;

内连接查询注意
1.从哪些表中查询数据
2.查询条件是什么
3.查询哪些字段
3.外连接查询
左外连接:
语法:select 字段列表 from 表1 left [outer] join 表2 on 条件;
查询的是左表所有数据及其交集
右外连接
语法:select 字段列表 from 表1 right[outer] join 表2 on 条件;
查询的是右表所有数据及其交集

-- 左外连接
SELECT * FROM dept;
SELECT * FROM emp;
-- 新入职员工,无任何部门,外键可以为空
-- 查询所有员工信息,如果员工有部门,则查询部门的名称,没有部门则不显示部门名称
-- 内连接
SELECT
t1.*,t2.`name`
FROM
emp t1,dept t2
WHERE
t1.`dept_id`=t2.`id`;
-- 白龙马查询不到
-- 利用左外连接显示白龙马
SELECT t1.*,t2.name FROM emp t1 LEFT JOIN dept t2 ON t1.`dept_id`=t2.`id`;
-- 右外连接
SELECT t1.*,t2.name FROM emp t1 RIGHT JOIN dept t2 ON t1.`dept_id`=t2.`id`;
-- 右外连接2ok
SELECT t1.*,t2.name FROM dept t2 RIGHT JOIN emp t1 ON t1.`dept_id`=t2.`id`;

4.子查询
4.1概念:查询中嵌套查询,称嵌套查询为子查询

-- 查询工资最高的员工信息
-- 1.查询最高的工资是多少9000
SELECT MAX(salary) FROM emp;
-- 2.查询员工信息,并且工资等于9000的
SELECT * FROM emp WHERE emp.`salary`=9000;
-- 一条SQL完成此操作
SELECT * FROM emp WHERE emp.`salary`=(SELECT MAX(salary) FROM emp);

4.2 子查询的不同情况

子查询的结果是单行单列的
子查询可以作为条件,使用运算符去判断。运算符:> >= < <= =
查询平均工资

-- 子查询的结果是单行单列的时候
-- 查询员工工资小于平均工资的人 4680
SELECT * FROM emp WHERE emp.salary < (SELECT AVG(salary) FROM emp);

子查询的结果是多行单列的
查询的结果是多行单列的,可以使用运算符in来判断

-- 子查询,查询的结果是多行单列的,可以使用运算符in来判断
-- 查询财务部和市场部所有员工的信息
SELECT id FROM dept WHERE NAME='财务部' OR NAME='市场部';
SELECT * FROM emp WHERE dept_id=3 OR dept_id=2;
-- 合并为一条SQL
-- 多个字段or,可以简化为in
SELECT * FROM emp WHERE dept_id IN (2,3);
SELECT * FROM emp WHERE dept_id IN (SELECT id FROM dept WHERE NAME='财务部' OR NAME='市场部');

子查询的结果是多行多列的
查询员工的入职日期为2011-11-11之后的员工信息和部门信息
多行多列的,子查询可以作为一张虚拟表参与表的查询

-- 子查询
-- 查询员工的入职日期为2011-11-11之后的员工信息和部门信息
SELECT * FROM emp WHERE emp.`join_date` > '2011-11-11';
-- 还要查询部门信息,上表和部门信息做关联
SELECT * FROM dept t1, (SELECT * FROM emp WHERE emp.`join_date` > '2011-11-11') t2
WHERE t1.id=t2.id;
-- 普通内连接
SELECT * FROM emp t1,dept t2 WHERE t1.`dept_id`=t2.`id` AND t1.`join_date` > '2011-11-11';

5.多表查询-练习

-- 需求:

-- 1.查询所有员工信息。查询员工编号,员工姓名,工资,职务名称,职务描述

/*
分析:
1.员工编号,员工姓名,工资属于emp表,职务名称,职务描述属于job表
2.查询条件:emp.job_id=job.id
*/
SELECT
t1.`id`,-- 员工编号
t1.`ename`,-- 员工姓名
t1.`salary`,-- 员工工资
t2.`description`,-- 职务描述
t2.`jname`-- 职务名称
FROM
emp t1,
job t2
WHERE
t1.`job_id`=t2.`id`;

-- 2.查询员工编号,员工姓名,工资,职务名称,职务描述,部门名称,部门位置

/*
分析:
1.员工编号,员工姓名,工资 查询自 emp,职务名称,职务描述 查询自 job,部门名称,部门位置 查询自 dept
2.条件:emp.job_id=jpb.id AND emp.dept_id=dept.id
*/
SELECT
t1.`id`,-- 员工编号
t1.`ename`,-- 员工姓名
t1.`salary`,-- 员工工资
t2.`description`,-- 职务描述
t2.`jname`,-- 职务名称
t3.`dname`,-- 部门名称
t3.`loc`-- 部门位置 FROM
emp t1,
job t2,
dept t3
WHERE
t1.`job_id`=t2.`id` AND t1.`dept_id`=t3.`id`;

-- 3.查询员工姓名,工资,工资等级

/*
分析:
1.员工姓名,工资 emp 工资等级 salarygrade
2.条件 emp.salary >= salarygrade.losalary and emp.salary <= salarygrade.hisalary;
或 emp.salary between salarygrade.losalary and salarygrade.hisalary;
*/
SELECT
t1.`ename`,
t1.`salary`,
t2.*
FROM emp t1,salarygrade t2
WHERE t1.`salary` BETWEEN t2.`losalary` AND t2.`hisalary`;

-- 4.查询员工姓名,工资,职务名称,职务描述,部门名称,部门位置,工资等级

/*
分析:
1.员工姓名,工资 emp,职务名称,职务描述 job,部门名称,部门位置 dept,工资等级 salarygrade
条件emp.job_id=jpb.id AND emp.dept_id=dept.id AND emp.salary between salarygrade.losalary and salarygrade.hisalary
*/
SELECT
t1.`ename`,t1.`salary`,t2.`jname`,t2.`description`,t3.`dname`,t3.`loc`,t4.`grade`
FROM
emp t1, job t2, dept t3, salarygrade t4
WHERE
t1.`job_id`=t2.`id`
AND t1.`dept_id`=t3.`id`
AND t1.`salary` BETWEEN t4.`losalary` AND t4.`hisalary`;

-- 5.查询出部门编号、部门名称、部门位置、部门人数【分组查询和子查询】

/*
分析:
1.部门编号、部门名称、部门位置 dept表 部门人数 emp表
2.如何查询部门的人数?使用分组查询,按照emp.dep_id完成分组,查询count(id)
3.使用子查询将第2步的查询结果和dept表进行关联查询
*/
SELECT
t1.`id`,t1.`dname`,t1.`loc`,t2.total
FROM
dept t1,(SELECT dept_id,COUNT(id) total
FROM emp
GROUP BY dept_id) t2
WHERE t1.`id`=t2.dept_id;-- 将其当做虚拟表,和部门表关联

-- 6.查询所有员工的姓名及其直接上级的姓名,没有领导的员工也需要查询

/*
分析:
1.姓名 emp, 直接上级的姓名 emp
emp表的id和mgr 是自关联的【两张表】
2.条件 emp.id=emp.mgr
3.查询左表的所有数据和交集数据
使用左外连接查询
*/
/*
select
t1.`ename`,
t1.`mgr`,
t2.`id`,
t2.`ename`
from emp t1 ,emp t2
where t1.id=t2.`mgr`;
*/
-- 查询罗贯中,唐僧等没有上级的,使用左外连接
SELECT
t1.`ename`,
t2.`mgr`,
t2.`id`,
t2.`ename`
FROM emp t1
LEFT JOIN emp t2
ON t1.`mgr`=t2.`id`

二、事务
1.基本的基本介绍
1.1 概念:
            如果一个包含多个步骤的业务操作,被事务管理,那么这些操作要么同时成功,要么同时失败
      例子:张三给李四转账500元

查询张三账户余额是否大于500
张三账户金额  -500
李四账户金额 +500
不用事务管理,则中途失败
被事务管理,出现异常,则会回滚
事务的操作:开启事务、回滚、提交
开启事务:start transaction
回滚:rollback
提交:commit

-- 创建数据表
CREATE TABLE account (
id INT PRIMARY KEY AUTO_INCREMENT,
NAME VARCHAR(10),
balance DOUBLE
);
-- 添加数据
INSERT INTO account (NAME, balance) VALUES ('zhangsan', 1000), ('lisi', 1000);
-- 0 开启事务
START TRANSACTION;
-- 1张三给李四转账500元
-- 2查询张三账户余额是否大于500
-- 3张三账户-500
UPDATE account SET balance = balance - 500 WHERE NAME='zhangsan';
-- 李四账户+500
-- 出错了
UPDATE account SET balance =balance + 500 WHERE NAME='lisi';
SELECT * FROM account;-- 临时数据的变化,不是持久的变化
-- 发现执行没有问题,提交事务
COMMIT;
-- 发现出问题了,应该 回滚事务
ROLLBACK;-- 回滚,保证账户的安全性

1.2 MySQL数据库中,事务默认提交

事务提交的两种方式
自动提交:
MySQL就是自动提交的
一条DML(增删改)语句会自动提交一次事务
手动提交
需要先开启事务,再提交

-- 自动提交
UPDATE account SET balance=1000;
-- 手动提交
COMMIT; -- 不自动提交,事务会默认回滚

修改事务的默认提交方式【MySQL是自动提交的,Oracle是手动提交的,必须写commit才可以生效】
查看事务的默认提交方式 SELECT @@autocommit; --1代表自动提交,0代表手动提交
修改默认提交方式:set @@autocommit=0;

SELECT @@autocommit; -- 1代表自动提交,0代表手动提交
SET @@autocommit=0; -- 必须提交,才能正确
UPDATE account SET balance =30;
COMMIT;

2.事务的四大特征(常见的面试题)

原子性:不可分割的最小操作单位,要么同时成功,要么同时失败
持久性:事务一旦提交/回滚,数据库会持久化的保存数据
隔离性:多个事务之间,相互独立。
一致性:事务操作前后,数据总量不变

3.事务的隔离级别(了解)
3.1 概念

隔离性:多个事务之间是隔离的,相互独立的。但是如果多个事务操作同一批数据,则会引发一些问题,设置不同的隔离级别,就可以解决这些问题。
存在的问题:
脏读:一个事务,读取到另一个事务中没有提交的数据
不可重复读(虚读):在同一个事务中,两次读取到的数据不一样

幻读:一个事务操作(DML)数据表中所有的记录,另一个事务添加了一条数据,则第一个事务查询不到自己的修改。
隔离级别【一般只有特殊需求才需要修改,了解即可】
read uncommitted:读未提交
会产生的问题:脏读、不可重复读、幻读
read committed:读已提交(只有提交了数据,另一个事务才能读到)(Oracle默认)
会产生的问题 :不可重复读、幻读
repeatable read:可重复读(MySQL默认)
会产生的问题:幻读
serializable:串行化(hang)【锁表,效率低】
可以解决所有的问题
注意:隔离级别从小大大,安全性越来越高,但是效率越来越低
数据库如何设置隔离级别
查询隔离级别 select @@tx_isolation;
设置隔离级别 set global transaction isolation level 级别字符串;

SELECT @@tx_isolation; -- mysql默认可重复读
-- Oracle默认读已提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;

3.2 演示

设置窗口的隔离级别为:读未提交 
SET GLOBAL TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
start transaction;
-- 转账操作,更新张三 和 李四 
update account set balance =balance -500 where  id=1;
update account set balance =balance +500 where  id=2;
3.3 事务的隔离级别--可重复读和串行化

三、DCL
1.概述
SQL分类:

DDL:增删改数据库和表
DML:增删改表中的数据
DQL:查询表中的数据
DCL:管理用户,授权

DBA:数据库管理员
2.对用户的管理操作

DCL:管理用户,授权
管理用户
添加用户:CREATE USER '用户名@主机名' identify 密码;

-- 1.切换到MySQL数据库
USE mysql;
-- 2.查询user表
SELECT * FROM USER;
-- 两个root分别表示本地数据库和远程数据库 -- 创建用户
CREATE USER '用户名@主机名' IDENTIFIED BY 密码;
CREATE USER 'zhangsan'@'localhost' IDENTIFIED BY '123';
SELECT * FROM USER;
-- 任意电脑都可访问 CREATE USER 'lisi'@'%' IDENTIFIED BY '123';

删除用户

CREATE USER 'lisi'@'%' IDENTIFIED BY '123';

-- 删除用户
DROP USER '用户名'@'主机名';
DROP USER 'zhangsan'@'localhost';

修改用户密码

-- 修改lisi用户密码为abc
UPDATE USER SET PASSWORD=PASSWORD('新密码') WHERE USER='用户名';
SELECT * FROM USER; UPDATE USER SET PASSWORD=PASSWORD('123') WHERE USER='lisi'; -- DCL中的简化书写形式【DCL特有的方式】
SET PASSWORD FOR '用户名'@'主机名' = PASSWORD('密码');
SET PASSWORD FOR 'root'@'localhost' = PASSWORD('root');

忘记密码的解决方案
停止MySQL的服务(需要管理员运行该cmd):net stop mysql
使用无验证方式启动mysql服务:mysqld --skip-grant-tables
在另一个cmd里面直接输入mysql
改mysql密码:update user set password=password('root') where user='root'
进程管理器结束mysqld的进程
cmd启动mysql服务:net start mysql
重新登录
查询用户

-- 1.切换到MySQL数据库
USE mysql;
-- 2.查询user表
SELECT * FROM USER;
-- 两个root分别表示本地数据库和远程数据库

百分号表示通配符,可以在任意 计算机上登录

授权

3.权限管理

查询权限

-- 查询权限
SHOW GRANTS FOR '用户名'@'主机名';
SHOW GRANTS FOR 'lisi'@'%';-- usage 只能查询
SHOW GRANTS FOR 'root'@'%';-- root有各种权限

授予权限

-- 授予权限
GRANT 权限列表 ON 数据库名.表名 TO '用户名'@'主机名';
GRANT SELECT, DELETE , UPDATE ON db3.account TO 'lisi'@'%'; -- 给张三用户授予所有权限,在任意数据库的任意表上
-- 通配符all表示所有权限
-- 通配符 *.*表示所有数据库的所有表
GRANT ALL ON *.* TO 'zhangsan'@'localhost';

撤销权限

-- 撤销权限
REVOKE 权限列表 ON 数据库.表名 FROM '用户名'@'主机名';
REVOKE UPDATE ON db3.`account` FROM 'lisi'@'%';

来自为知笔记(Wiz)

JavaEE Day04 MySQL多表&事务的相关教程结束。

《JavaEE Day04 MySQL多表&事务.doc》

下载本文的Word格式文档,以方便收藏与打印。