Hi,大家好,我是编程小6,很荣幸遇见你,我把这些年在开发过程中遇到的问题或想法写出来,今天说一说一份超级详细的Mysql数据库笔记(值得收藏)「建议收藏」,希望能够帮助你!!!。
mysql服务启动,在cmd输入net start mysql
#创建数据库
CREATE DATABASE hsp_db01;
#创建一个使用 utf8 字符集的 hsp_db02 数据库
CREATE DATABASE hsp_db02 CHARACTER SET utf8
#创建一个使用 utf8 字符集,并带校对规则的 hsp_db03 数据库
CREATE DATABASE hsp_db03 CHARACTER SET utf8 COLLATE utf8_bin
#校对规则 utf8_bin 区分大小 默认 utf8_general_ci 不区分大小写
#查看当前数据库服务器中的所有数据库
SHOW DATABASES
#查看创建的 hsp_db01 数据库的定义信息
SHOW CREATE DATABASE `hsp_db01` #为了规避关键字,可以使用反引号
#删除前面创建的 hsp_db01 数据库
DROP DATABASE hsp_db01
mysqldump -u 用户名 -p 密码 -B 数据库名 > 路径(例:d:\\database.sql)
备份数据库的多个表不能带-B,否则会被认为是多个数据库
mysqldump -u 用户名 -p 密码 数据库名 表1 表2 > 路径(例:d:\\database.sql)
使用mysqldump命令来恢复一个叫data01数据库的一个叫table01的表(已有备份文件)。具体步骤如下:
先进入mysql命令行:mysql -u 用户名 -p 密码
恢复:source 文件路径.sql(例:d:\\database.sql)
左上角写了mysql -u 用户名 -p 的界面才叫mysql命令行
先用mysqldump -u 用户名 -p 密码 -B 数据库名 > 路径(例:d:\\database.sql)
从数据库里备份一个到E盘里,然后在数据库中用drop指令将其删除
然后通过mysql -u 用户名 -p 密码进入mysql命令行,输入source d:\\database.sql将其备份
decimal(M,D)中的M默认为10(最大65),D默认为0(最大30),M是位数,D是小数点后有几位
如果没有指定 unsinged , 就是有符号
#创建一个无符号的tinyint类型数据,0~255
CREATE TABLE t4 (id TINYINT UNSIGNED)
VARCHAR虽然可以有65535个字节,但是首先去掉3个字节用于记录字段大小,所以最大为65532个字节,
如果用的编码是UTF-8的话,一个字符消耗3个字节,最多存放65532 / 3 = 21844个字符,
GBK是2个字节,最多存放65532 / 2 = 32766个字符。
#我们来创建一个数据类型为varchar,编码为GBK的表
CREATE TABLE t10 (`name` VARCHAR(32766)) CHARSET gbk;
DATE / DATETIME / TIMESTAMP
CREATE TABLE t14 (
birthday DATE , -- 年月日
job_time DATETIME, -- 年月日 时分秒
login_time TIMESTAMP -- 如果希望 login_time 列自动更新, 需要配置这两句
NOT NULL DEFAULT CURRENT_TIMESTAMP -- 不允许为空,默认为当前时间
ON UPDATE CURRENT_TIMESTAMP -- 更新当前时间戳
);
CREATE TABLE `emp` (
id INT,
`name` VARCHAR(32),
sex CHAR(1),
brithday DATE,
entry_date DATETIME,
job VARCHAR(32),
salary DOUBLE,
`resume` TEXT
)
CHARSET utf8 COLLATE utf8_bin ENGINE INNODB
#charset是character set的简写,即字符集
#字符集,校验规则,存储引擎不写的话就是默认的(默认的话跟随当前数据库)
要用cmd在一个叫database的数据库下新建一个表,你需要先用mysql命令登录到数据库,然后用use命令选择database数据库,再用create table命令创建表。具体的步骤如下:
DROP TABLE `emp`
举例演示:
ALTER TABLE emp
-- 员工表 emp 的上增加一个 image 列,varchar 类型(要求在 birthday 后面)。
ADD image VARCHAR(32) NOT NULL DEFAULT '' AFTER birthday ;
-- DEFAULT ''默认为''这个字符
-- 修改 job 列,使其长度为 60。
ALTER TABLE emp
MODIFY job VARCHAR(60) NOT NULL DEFAULT '' ;
-- 删除 sex 列。
ALTER TABLE emp
DROP sex;
-- 表名改为 employee。
RENAME TABLE emp TO employee;
-- 修改表的字符集为 utf8
ALTER TABLE emp CHARACTER SET utf8;
-- 列名 name 修改为 user_name
ALTER TABLE emp;
-- 显示表结构,可以查看表的所有列
DESC emp ;
用法:
INSERT INTO 表名 (列名1,列名2...)
values (对应列名的值);
INSERT INTO `emp` (id, birthday)
VALUES(1, '2003-05-15'); -- 注意date等日期类型一定要按照规则写否则报错或者错位写入
#表里有3个数据,id,goodname,price有默认值10
INSERT INTO `goods` (id, goods_name)
VALUES(80, '格力手机');
#最后会插进去一个(80,格力手机,10)进去
用法:
UPDATE 表名 SET 列名 = 值 WHERE 布尔表达式,筛选满足条件的行
#将所有员工薪水修改为 5000 元。如果不带 where 条件,会修改该列所有的记录,因此要小心
UPDATE employee SET salary = 5000;
将姓名为 小妖怪 的员工薪水修改为 3000 元
UPDATE employee
SET salary = 3000
WHERE user_name = '小妖怪';
也可以修改多个列的值
UPDATE employee
SET salary = salary + 1000 , job = '出主意的'
WHERE user_name;
用法:
DELETE FROM 表名
where 布尔等值式; -- 如果不带 where 条件,会修改所有记录,小心删库跑路
#删除表中名称为’老妖怪’的记录。
DELETE FROM employee
WHERE user_name = '老妖怪';
SELECT [DISTINCT] * 或 指定列 FROM 表名;
-- distinct功能是去重(写在from前的每一个列的值都相同才去)
-- *代表所有列,也可以写多个列(用逗号隔开)
比如:
-- 查询表中所有学生的信息
SELECT * FROM student;
-- 查询表中所有学生的姓名和对应的英语成绩
SELECT
name
,english FROM student;
name
-- 过滤表中重复数据
SELECT DISTINCT
name
,english FROM student;
name
比如:
-- 统计每个学生的总分
SELECT `name`, (chinese + english + math ) FROM student;
#但是使用这种方法系统会自动命名第二列为chinese + english + math,所以我们可以采用下面的方式美化一下
-- 使用AS给列命名
SELECT
name
AS '名字', (chinese + english + math ) AS total
FROM student;
name
SELECT [DISTINCT] * 或 指定列 FROM 表名 WHERE 布尔表达式; -- select定最后显示多少列,from定从哪个表查询,where定最后显示满足条件的行
比如:
-- 查询姓名为赵云的学生成绩
SELECT * FROM student
WHERE
name
= '赵云';
-- 查询 math 大于 60 并且(and) id 大于 4 的学生成绩
SELECT * FROM student
WHERE math >60 AND id > 4;
-- 查询英语成绩大于语文成绩的同学
SELECT * FROM student
WHERE english > chinese;
-- 查询总分大于 200 分 并且 数学成绩小于语文成绩 并且 姓赵的学生, 赵% 表示名字以赵开头
SELECT * FROM student
WHERE (chinese + english + math) > 200 AND
math < chinese AND
name
LIKE '赵%';
-- 查询英语分数在 80-90 之间的同学。
SELECT * FROM student
WHERE english >= 80 AND english <= 90;
SELECT * FROM student
WHERE english BETWEEN 80 AND 90; -- between .. and .. 是 闭区间
name
name
-- 查询数学分数为 89,90,91 的同学。
SELECT * FROM student
WHERE math = 89 OR math = 90 OR math = 91;
SELECT * FROM student
WHERE math IN (89,90,91);
SELECT * FROM 表名 ORDER BY 列名(排序的依据) [DESC]; -- 默认为升序,DESC改为降序
比如:
-- 对数学成绩排序后输出【升序】。
SELECT *
FROM student
ORDER BY math;
-- 对总分按从高到低的顺序输出 [降序] -- 使用别名排序
SELECT
name
, (chinese + english + math) AS '总分'
FROM student
ORDER BY total_score DESC;
name
-- 对姓韩的学生成绩[总分]排序输出(升序) where + order by
SELECT
name
, (chinese + english + math) AS '总分' FROM student
WHERE
name
LIKE '韩%'
ORDER BY total_score;
name
name
比如:
-- % 表示 0 到多个的任意字符
-- _ 表示单个任意字符 其中GBK中需要两个_ _来表达一个字符
-- 显示首字符为 S 的员工姓名和工资
SELECT name, salary FROM emp
WHERE name LIKE 'S%'
-- 显示第三个字符为大写 O 的所有员工的姓名和工资
SELECT name, salary FROM emp
WHERE name LIKE '__O%'
DESC 表名
select * from table limit 参数1,参数2; -- 参数1 : 代表偏移量offset(从第offset + 1数据开始查),注意是从0开始,所以应该+1代表第一个数据
-- 参数2 : 取出的数据条数rows
/* 查询第1-10条数据 */
SELECT * FROM Student LIMIT 0,10;
/* 查询第11-20条数据 */
SELECT * FROM Student LIMIT 10 , 10; -- 讲到这我们会想知道如何快速算出 参数1,接下来我们看一下计算公式
公式
在进行分页之前,要注意一下是否超出了最大页数,所以我们需要先根据数据总量来得出总页数,这需要用到统计函数COUNT和向上取整函数CEIL,SQL操作如下:
#获得数据总条数
SELECT COUNT(*) FROM Student;
#假设每页显示10条,则直接进行除法运算,然后向上取整,这样我们就获得了总页数,在实际操作中,要注意一下是否超出了最大页数
SELECT CEIL(COUNT(*) / 10) AS '总页数' FROM Student;
多表查询的条件不能少于 要查询的表个数 - 1, 否则会出现笛卡尔集
自连接的特点
1. 把同一张表当做两张表使用
2. 需要给表取别名 如果from同一张表两次,会报错,所以需要对表重命名,无需使用 AS,直接表加别名就好
3. 列名有时不明确,可以通过 AS 指定列的别名
举例:
显示公司员工名字和他上级的名字,员工名字和上级的名字都在emp表
其中员工和上级是通过 emp 表的 manager列 关联
SELECT worker.name AS '职员名' ,
boss.name AS '上级名'
FROM emp worker, emp boss
WHERE worker.manager = boss.empno
#多行子查询中使用in运算符
#查询和10号部门的工作相同的雇员的 名字、岗位、工资、部门号, 但是不含10号部门的员工
select ename, job, sal, deptno
from emp
where job in (
SELECT DISTINCT job
FROM emp
WHERE deptno = 10
) and deptno != 10
多行子查询中使用 all 操作符
查询工资比部门 30 的所有员工的工资高的员工的姓名、工资和部门号
SELECT ename, sal, deptno
FROM emp
WHERE sal > ALL(
SELECT sal
FROM emp
WHERE deptno = 30
)
多行子查询中使用 any 操作符
查询工资比部门 30 的任意/其中一个员工的工资高的员工的姓名、工资和部门号
SELECT ename, sal, deptno
FROM emp
WHERE sal > any(
SELECT sal
FROM emp
WHERE deptno = 30
)
通常用于将两个关系表中具有相同属性的行组合在一起,以便我们可以从两个表中同时获取信息。
自然连接是一种特殊的等值连接(区别在于等值连接需要指定值),他要求两个表中进行连接的必须是相同的属性列(名字相同),无须添加连接条件,并且在结果中消除重复的属性列。
natural表1:
natural表2:
select 列名 from 左表 natural join 右表
-- 将2个表自然连接一下
SELECT * FROM `natural` NATURAL JOIN `natural2`
说人话,直接先把两张表的列名去重后列出来
然后开始对比他们两相同的列(id)的值,我们就找到id = 2时相同,直接把左表和右表连接起来,结果:
内连接基本与自然连接相同,不同之处在于自然连接的是同名属性列的连接,而内连接则不要求两属性列同名,可以用using或on来指定某两列字段相同的连接条件。
select 列名 from 左表 inner join 右表 on 条件 -- inner可以省略
通常用于查询在一张表中存在但在另一张表中不存在的记录
SELECT COUNT(*) 或 (列)FROM 表名 WHERE 表达式;
-- count(*) 和 count(列) 的区别
-- count(*) 返回满足条件的行数
-- count(列): 统计满足条件的某列个数,会排除为 null 的
SELECT SUM(列) FROM 表名 WHERE 表达式;
比如:
-- 统计一个班级语文、英语、数学各科的总成绩并使用别名
SELECT SUM(math) AS math_total_score,
SUM(english) AS english_total_score,
SUM(chinese) AS chinese_total_score
FROM student;
-- 统计一个班级语文成绩平均分
SELECT SUM(chinese)/ COUNT(*) FROM student;
-- 注意有些数据类型不能统计比如字符串之类
SELECT SUM(
name
) FROM student; #不报错但无意义
name
SELECT AVG(列) FROM 表名 WHERE 表达式;
比如:
-- 求一个班级数学平均分?
SELECT AVG(math) FROM student;
-- 求一个班级总分平均分
SELECT AVG(math + english + chinese) FROM student;
-- 求班级最高分和最低分
SELECT MAX(math + english + chinese), MIN(math + english + chinese)
FROM student;
-- 求出班级数学最高分和最低分并使用别名
SELECT MAX(math) AS math_high_socre,
MIN(math) AS math_low_socre
FROM student;
注意:当聚合列和非聚合列出现在一起时必须使用group by语句
SELECT 列名 FROM 表名 group by 列名(从select后面的几个列名之中挑);
比如:
-- 显示每个部门的每种岗位的平均工资和最低工资
SELECT AVG(salary), MIN(salary) , department, job
FROM emp GROUP BY department, job;
-- GROUP BY也可以对查询结果进行去重
SELECT 列名 FROM 表名 group by 列名 having 布尔表达式;
比如:
-- 显示平均工资低于 2000 的部门编号和它的平均工资 并使用别名
SELECT AVG(salary) AS '平均工资',
deptno AS '部门编号'
FROM emp
GROUP BY deptno
HAVING AVG(salary) < 2000; # 注意HAVING 子句中不能使用别名,只能使用原始的表达式
MySQL有一些加密函数,可以用来对字符串进行加密和解密。常用的有以下几种:
AES_ENCRYPT(str,key):返回用密钥key对字符串str利用高级加密标准算法加密后的结果,调用AES_ENCRYPT的结果是一个二进制字符串,以BLOB类型存储。
AES_DECRYPT(crypt_str,key):返回用密钥key对字符串crypt_str利用高级加密标准算法解密后的结果1。
DECODE(str,key):使用key作为密钥解密加密字符串str2。
ENCODE(str,key):使用key作为密钥加密字符串str2。
MD5(str):返回str的MD5校验和,是一个32位十六进制数2。
演示MD5使用
-- 插入
INSERT INTO users (username, password) VALUES ('admin', MD5('123456'));
-- 这样就可以把用户名和密码加密后存储在users表中。你也可以使用MD5函数来验证用户输入的密码是否正确,例如:
SELECT * FROM users WHERE username = 'admin' AND password = MD5('123456');
这样就可以根据用户名和密码查询users表中是否有匹配的记录。你明白了吗?
MySQL约束语句是用来限制表中数据的条件。MySQL支持以下几种约束:
可以在创建表或修改表时添加或删除约束。
我来介绍一下这6种约束的用法。
不能重复而且不能为 null
在创建表时,可以使用PRIMARY KEY关键字来指定一个或多个列作为主键,例如:
CREATE TABLE student (
id INT PRIMARY KEY,
name VARCHAR(20),
age INT
);
#复合主键
CREATE TABLE t(
id INT ,
`name` VARCHAR(32),
PRIMARY KEY (id, `name`) -- 这里就是复合主键
);
也可以在创建表后,使用ALTER TABLE语句来添加或删除主键,例如:
ALTER TABLE student ADD PRIMARY KEY (id);
ALTER TABLE student DROP PRIMARY KEY;
外键是用于建立和加强两个表数据之间的链接的列。 外键的作用是保持数据的一致性、完整性,主要体现在下面两个方面
在创建表时,可以使用FOREIGN KEY关键字来指定一个或多个列作为外键,并引用主表中的主键(primary key)或唯一键(unique)
用法:FOREIGN KEY (外键所在表的外键列列名) REFERENCES 主表名(主表中的主键或唯一键)
比如:
-- 创建 主表 my_class (外键指向表)
CREATE TABLE my_class (
id INT PRIMARY KEY , -- 班级编号
name
VARCHAR(32) NOT NULL DEFAULT ''
);
name
-- 创建 从表 my_stu (外键所在表)
CREATE TABLE my_stu (
id INT PRIMARY KEY , -- 学生编号
name
VARCHAR(32) NOT NULL DEFAULT '',
class_id INT , -- 学生所在班级的编号
FOREIGN KEY (class_id) REFERENCES my_class(id) -- 外键
);
name
CREATE TABLE user (
id INT PRIMARY KEY,
username VARCHAR(20) UNIQUE NOT NULL, -- unique + not null = primary key
password VARCHAR(20) NOT NULL
);
在创建表时,可以使用NOT NULL关键字来指定一个列的值不为空,例如:
CREATE TABLE book (
id INT PRIMARY KEY,
title VARCHAR(50) NOT NULL,
);
CREATE TABLE employee (
id INT PRIMARY KEY,
name VARCHAR(20) NOT NULL,
salary DECIMAL(10,2),
CHECK (salary >0)
);
CREATE TABLE post (
id INT PRIMARY KEY,
title VARCHAR(50) NOT NULL,
content TEXT,
create_time DATETIME DEFAULT CURRENT_TIMESTAMP
);
目的:当我们希望一列每插一个数据进去会自动增长时可以使用自增长,这样可以避免手动输入主键值,也可以保证主键值的唯一性
要实现MySQL自增长,需要满足以下条件:
用法:
字段名 整型 primary key auto_increment
-- 创建一个名为student的表,包含id、name和age三个字段
CREATE TABLE student (
id INT NOT NULL AUTO_INCREMENT, -- id字段为自增长字段,必须为整数类型且非空
name VARCHAR(20) NOT NULL, -- name字段为字符串类型,非空
age INT NOT NULL, -- age字段为整数类型,非空
PRIMARY KEY (id) -- id字段为主键,保证唯一性
);
-- 插入两条记录,不需要指定id值,数据库会自动赋值
INSERT INTO student (name, age) VALUES ('Alice', 18);
INSERT INTO student (name, age) VALUES ('Bob', 19);
-- 查询表中的数据,可以看到id值自动递增了
SELECT * FROM student;
输出结果如下:
id name age
1 Alice 18
2 Bob 19
-- 如果插入的时候插入一个不是自增长的值时,再插入数据则从上一个开始自增
INSERT INTO student (id ,name, age) VALUES (666,'Tom', 18);
INSERT INTO student (name, age) VALUES ('John', 18);
-- 查询
SELECT * FROM student;
输出结果如下:
id name age
1 Alice 18
2 Bob 19
666 TOM 18
667 John 18
MySQL索引是一种帮助MySQL高效获取数据的数据结构。它可以根据一列或多列的值进行排序,从而加快数据库的查询速度
MySQL中常用的索引结构有B-TREE,B+TREE,HASH等。不同的索引结构适用于不同的查询场景,例如:
用法:
CREATE INDEX 索引名 ON 表名 (列名);
ALTER TABLE 表名 ADD INDEX 索引名 (列名);
在创建表时,可以使用 PRIMARY KEY,UNIQUE或INDEX关键字来定义主键、唯一或普通索引12。例如:
-- 创建一个名为student的表,包含id、name和age三个字段
CREATE TABLE student (
id INT NOT NULL AUTO_INCREMENT PRIMARY KEY, -- id字段为主键索引
name VARCHAR(20) NOT NULL UNIQUE, -- name字段为唯一索引
age INT NOT NULL INDEX -- age字段为普通索引
);
在已有的表上,可以使用 ALTER TABLE 或 CREATE INDEX语句来创建索引
-- 在student表上创建一个名为idx_name_age的复合索引,包含name和age两个字段
ALTER TABLE student ADD INDEX idx_name_age (name, age);
-- 或者
CREATE INDEX idx_name_age ON student (name, age);
单独创建主键索引,例如:
ALTER TABLE student ADD PRIMARY KEY (id);
删除主键索引,例如:
ALTER TABLE student DROP PRIMARY KEY;
在已有的表上,可以使用 DROP INDEX 或 ALTER TABLE语句来删除索引。
-- 删除student表上的idx_name_age索引
DROP INDEX idx_name_age ON student;
-- 或者
ALTER TABLE student DROP INDEX idx_name_age;
事务是指一组操作量大,复杂度高的数据操作,它们要么都执行成功,要么都执行失败,不会出现中间状态。
MySQL默认每条DML(增删改)操作都是一个事务。可以通过 SHOW VARIABLES LIKE 'autocommit’ 查看自动提交模式是否开启。如果关闭自动提交模式,则需要手动使用COMMIT或ROLLBACK来提交或回滚事务
MySQL只有使用了InnoDB数据库引擎的数据库或表才支持事务。
START TRANSACTION; / SET autocommit = off; -- 开启一个新的事务
SAVEPOINT 保存点名; -- 设置保存点
ROLLBACK TO 保存点名; -- 回滚到某事务,
-- (注意)假如有按顺序的3个保存点A,B,C 如果回滚到A时,B和C保存点就被删除
ROLLBACK; -- 回滚到最开始的状态
COMMIT; -- 提交当前事务,一旦执行,就不能再回滚了,保存点也被删除,也只有提交事务之后,其他客户端才(可能)看见数据库的更新,为什么是可能,引用到后面的隔离级别知识点
MySQL事务的隔离级别。事务的隔离级别是指多个并发事务之间如何相互影响的问题,不同的隔离级别会导致不同的并发问题。
那么脏读、不可重复读、幻读都是什么意思呢?
在MySQL中,可以使用以下命令设置事务的隔离级别:
set session transaction isolation level 隔离级别; -- read committed /read uncommitted / repeatable read / serializable
例如,设置隔离级别为读取未提交:
set session transaction isolation level read uncommitted;
设置隔离级别为读取已提交:
set session transaction isolation level read committed;
你可以使用以下语句查看当前数据库的隔离级别:
SELECT @@global.tx_isolation; -- 查看全局事务隔离级别
SELECT @@tx_isolation; -- 会话事务当前级别
用法: CREATE VIEW 视图名 AS
SELECT 列名... FROM 表名(基表);
-- 创建一个视图 emp_view01,只能查询 emp 表的(empno、ename, job 和 deptno ) 信息
CREATE VIEW emp_view01 AS
SELECT empno, ename, job, deptno FROM emp;
-- 1. 创建视图后,到数据库去看,对应视图只有一个视图结构文件(形式: 视图名.frm)
-- 2. 视图的数据变化会影响到基表,基表的数据变化也会影响到视图[insert update delete]
#创建用户
CREATE USER '用户名' @ '允许登录的IP地址' IDENTIFIED BY '密码'
-- 创建用户 shunping 密码 123 , 从本地登录
CREATE USER 'shunping'@ 'localhost' IDENTIFIED BY '123'
删除用户
DROP USER '用户名'@ '登录地址'
DROP USER 'shunping'@ 'localhost'
修改密码
-- 改自己的
set password = password('密码');
-- 改别人的
set password for '用户名'@ '登录地址' = password('密码');
-- '用户名' @ '登录地址'也叫用户账户
给用户授权
GRANT 权限列表.. -- 比如增删改查等权限
ON 库.对象名.. -- 比如赋予对象访问student数据库的(表,视图等对象)的权限
TO 用户账户 [WITH GRANT OPTION]; -- 加上最后的[WITH.. ]代表允许该用户将权限授予给其他用户
-- 给 shunping 分配查看 (SELECT) test库下的 news表 的权限 和 添加 (INSERT) 的权限
GRANT SELECT , INSERT
ON test.news
TO 'shunping'@'localhost';
回收用户权限
REVOKE 权限列表
ON 库.对象
FROM 用户账户
-- 回收 shunping 用户在 testdb.news 表的增改查权限
REVOKE SELECT , UPDATE, INSERT
ON testdb.news
FROM 'shunping'@'localhost'
-- 回收 shunping 用户在 testdb.news 表的所有权限
REVOKE ALL
ON testdb.news
FROM 'shunping'@'localhost
比如:
-- xxx 用户可以在任何ip登录 mysql
create user 'xxx';
-- xxx 用户在 192.168.1.*的 ip 可以登录 mysql
create user 'xxx'@ '192.168.1.%' ;
DROP USER jack -- 不填host默认就是 DROP USER 'jack'@ '%'