时间:2021-07-01 10:21:17 帮助过:30人阅读
目录
用户认证
只是通过文件操作,改变数据是非常繁琐的
解决对于多台机器或多个进程操作同一份数据我们需要自己解决并发和安全问题比较麻烦
自己处理数据备份,容错措施
c/s架构的操作数据文件的一个管理软件
数据操作
专有名词
常见的数据库
关系型数据库
非关系型数据库(key:value结构)eg:快递单号(redis、mongodb,memcache)
在 WEB 应用方面,MySQL是最好的 RDBMS(Relational Database Management System,关系数据库管理系统) 应用软件。
# 客户端
[mysql]
# 设置mysql客户端默认字符集
default-character-set=utf8
# server端
[mysqld]
#设置3306端口
port = 3306
# 设置mysql的安装目录
basedir=C:\Program Files\mysql-5.6.39-winx64
# 设置mysql数据库的数据的存放目录
datadir=C:\Program Files\mysql-5.6.39-winx64\data
# 允许最大连接数
max_connections=200
# 服务端使用的字符集默认为8比特编码的latin1字符集
character-set-server=utf8mb4
# 创建新表时将使用的默认存储引擎
default-storage-engine=INNODB
# windows
mysqld install
net start mysql
# server net stop mysql
mysql -u'用户名' -p'密码'
# 客户端,可以是python代码也可以是一个程序
# mysql.exe是一个客户端
# mac
sudo mysql.server status
sudo mysql.server start/stop/restart
Note:可以连接网络上的某一个数据库
sql语句(structure query language)
# 查看当前用户
select user();
# 设置密码,password 表示密文存储
set password = password('123');
# 创建用户
create user '用户名'@'网段.%' identified by '密码';
# 查看当前库
show databases;
# 创建文件夹henry
create database henry;
# 查看指定用户权限
show grants for '用户名'(@'网段.%');
# 授权 * 表示所有
grant all(select/insert) on henry.* to '用户名'@'ip网段.%';
# 设置立即生效
flush privileges
# 创建账号并授权,必须有密码
grant all on henry.* to 'henry'@'%' identified by '123';
#select, insert, update, delete, create, drop, index, alter, grant, references, reload, shutdown, process, file等14个权限
# 取消用户权限
revoke all on test.* from 'henry'@'%';
# 删除用户
delete from mysql.user where host='192.168.12.%' and user='test';
drop user 'test'@'192.168.12
# 修改指定用户密码
update mysql.user set password=password('新密码') where User='test' and Host='%';
# 创建库
create database demo;
# 查看库
show databases;
# 删除库,demo
drop database demo
# 查看当前使用的库
select database();
# 切换库,到demo库下
use demo;
# 创建表,char()默认一个字符
create table student(id int, name char(10));
# 查看当前文件夹中的表
show tables;
# 删除表
drop table student;
# 查看表结构
desc student;
# 删除多个表
drop tables s2,s3,s4;
# 数据插入
insert into student values(1, 'henry');
# 数据查看
select * from student;
# 修改数据,必须设置条件,确定为一条数据data
update 表 set 字段名=值 where id=2;
# 删除数据
delete from 表 where id=1;
其他存储引擎
# 查看与存储引擎相关配置
show variables like '%engine%';
show variables like "default_storage_engine";
# 查看当前数据库支持的存储引擎
show engines \g
show engines;
# 修改已经存在表的存储引擎
alter table 表名 engine = innodb;
# 查看与编码相关的配置
show variables like '%chara%';
# 查看
show variables like '%关键字%';
# 创建表
create table t1(id int, name char(10)) engine=innodb;
# 查看表的结构,包括存储引擎和编码 \G 格式化输出,带 \G 不能使用分号
show create table t1 \G
# 只查看表字段基础信息
describle t1;
t1.frm frame 表结构
t1.ibd innoDB 存储引擎
# 指定engine为myisam
create table t2(id int, name char(10)) engine=MyISAM;
t2.frm 表结构
t2.MYD 数据
t2.MYI 索引
# 指定engine为memory
create table t2(id int, name char(10)) engine=memory;
t2. 数据
# 语法:
create table 表名(
字段名1 类型[(宽度) 约束条件],
字段名2 类型[(宽度) 约束条件],
字段名3 类型[(宽度) 约束条件]);
# 注意:
1. 在同一张表中,字段名是不能相同
2. 宽度和约束条件可选
3. 字段名和类型是必须的
# 创建无符号int型表
create table t3(id1 int, id2 int unsigned);
# 一共有5位,小数2位
float(5, 2)/ double(5, 2)
# 创建表
create tables t4(f1 float(5,2), double(5,2));
# 不指定长度,单精度和双精度
create tables t4(f1 float, double);
# decimal精度,默认存储(10,0)整数
create table t5(d1 decimal, d2 decimal(25, 20));
# decimal内部存储是按照字符串存的
# 创建表
create table t6(d1 date, y year, ts timestamp);
insert into t6(now(), now(), now());
# 指定传y,datetime默认为更新时间
insert into t6(y) values(2019);
# 指定datetime更新方式
create table t6(d1 date, y year,
dt datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP);
# 可以使用字符串,纯数字
# 5.7版本,插入参数不全,会报错,5.6版不会
create table t7(name1 char(5), name2 varchar(5));
# 分别存储 'echo ' 和 'echo4'
insert into t7 values('echo', 'echo')
select concat(name1, '---') from t7;
select concat(name2, '---') from t7;
create table t8(name char(12),
gender ENUM('male', 'female'),
hobby set('play', 'drink', 'eat')s
);
create table t1(id int not null,
name char(12) not null,
age int
);
insert into t1(id, name) values(1, 'henry');
create table t2(id int not null,
name char(12) not null,
gender enum('male', 'female') not null default 'male'
);
insert into t2(id, name) values(1, 'henry');
create table t3(id int unique,
username char(12) not null unique,
pwd char(18)
);
create table t4(id int not null unique,
ip char(15),
server char(10),
port int,
unique(ip, port)) # 联合唯一,不能同时重复
create table t5(id int unique auto_increment,
username char(10),
pwd char(18));
insert into t5(username, pwd) values('henry', '123');
# 自增大小只增不减
# 对于自增的字段,在用delete删除后,再插入值,该字段仍按照删除前的位置继续增长
# 也可以创建表时指定auto_increment的初始值,注意初始值的设置为表选项,应该放到括号外
create table student(id int primary key auto_increment,
name varchar(20),
gender enum('male','female') default 'male'
)auto_increment=3;
#设置步长
# sqlserver:自增步长
# 基于表级别,指定步为2,从0开始计数
create table t1(id int unique auto_increment, age int
)engine=innodb,auto_increment=2 default charset=utf8;
# mysql自增的步长:
show session variables like 'auto_inc%';
# 基于会话级别
set session auto_increment_increment=2;# 修改会话级别的步长
# 基于全局级别的
set global auto_increment_increment=2; # 修改全局级别的步长(所有会话都生效)
# 查看设置,重新登陆有效,5.7版本直接失效
show variables like 'auto_incre%';
create table t6(id int not null unique,
name char(10) not null unique);
# 第一个指定为not null nuique 字段被定义为主键
create table t7(id int primary key,
name char(10) not null unique);
create table t8(id int,
ip char(15),
server char(10),
port int,
primary(ip, port)) # 联合主键
create table staff(id int primary key auto_increment,
age int,
gender enum('male', 'female'),
salary float(10,2),
hire_date date,
post_id int,
foreign key(post_id) references dept(pid);
create table dept(pid int primary key, name char(10) not null unique);
create table staff(id int primary key auto_increment,
age int,
gender enum('male', 'female'),
salary float(10,2),
hire_date date,
post_id int,
foreign key(post_id) references dept(pid)
on update cascade
on delete set null);
create table dept(pid int primary key, name char(10) not null unique);
# 修改表名
alter table 表名 rename 新表名;
# 添加字段
alter table 表名 add 添加字段名 数据类型(宽度) 约束
# 删除字段
alter table 表名 drop 删除字段名;
# 修改已经存在字段的类型、宽度 约束,不能修改字段名字
alter table 表名 modify 字段名 类型() 约束
# 修改已经存在字段的类型、宽度 约束、字段名字
alter table 表名 change 字段名 新字段名 类型() 约束
# 把字段放在第一列
alter table 表名 modify age 类型+约束 first;
# 把字段放在id之后
alter table 表名 modify age int not null after id;
# 也可以与 add、change 连用
#去掉null约束
alter table t modify name char(10) null;
# 添加null约束
alter table t modify name char(10) not null;
# 去掉unique约束,特殊
alter table 表名 drop index 字段名;
# 添加unique约束
alter table 表名 modify 字段名 int unique;
alter database 库名 CHARACTER SET utf8;
# 先删除主键,删除一个自增主键会报错
# 需要先去掉主键的自增约束,然后再删除主键约束
alter table 表名 drop primary key;
# 增加主键
alter table 表名 add primary key(id);
# 添加外键
alter table 表名 add constraint 外键名 foreign key(字段) references press(字段);
# 删除外键
alter table 表名 drop foreign key 外键名;
drop table 表名;
两张表的数据关系:多对一、一对一、多对多(书、作者)
create table press(id int primary key auto_increment,
name varchar(20));
create table book(id int primary key auto_increment,
name varchar(20),
press_id int not null,
foreign key(press_id) references press(id)
on delete cascade
on update cascade);
insert into press(name) values('henry publisher'),
('echo publisher'),('dean publisher');
insert into book(name,press_id) values('henry',1),('echo',2),
('dean',2),('brad',3),('dianel',2),('oleg',3);
# 两张表:学生表和客户表
create table customer(id int primary key auto_increment,
name varchar(20) not null,
qq varchar(10) not null,
phone char(16) not null);
create table student(id int primary key auto_increment,
class_name varchar(20) not null,
customer_id int unique, #该字段一定要是唯一的
foreign key(customer_id) references customer(id)
on delete cascade
on update cascade);
# 增加客户
insert into customer(name,qq,phone) values('henry', '12345', 12312312311), ('echo','123123123',12312312311),('dean', '283818181', 12312312311), ('brad','283818181',12312312311), ('oleg', '888818181', 12312312311), ('dianel','112312312',12312312311);
# 增加学生
insert into student(class_name,customer_id) values('1班',3),('2班',4),('3班',5);
create table author(id int primary key auto_increment,
name varchar(20));
create table book(id int primary key auto_increment,
name varchar(20));
# 这张表就存放作者表与书表的关系,即查询二者的关系查这表就可以了
create table author_book(id int not null unique auto_increment,
author_id int not null,
book_id int not null,
constraint fk_author foreign key(author_id) references author(id)
on delete cascade
on update cascade,
constraint fk_book foreign key(book_id) references book(id)
on delete cascade
on update cascade,
primary key(author_id,book_id));
# 插入作者和书籍信息
insert into author(name) values('henry'),('echo'),('dean'),('diane');
insert into book(name) values('1'),('2'),('3'),('4'),('5'),('6')
insert into author_book(author_id,book_id) values(1,1),(1,2),(1,3),(1,4),(1,5),(1,6),(2,1),(2,6),(3,4),(3,5),(3,6),(4,1);
# 在父表上update/delete记录时,同步update/delete掉子表的匹配记录
cascade方式
# 在父表上update/delete记录时,将子表上匹配记录的列设为null要注意子表的外键列不能为not null
set null方式
# 如果子表中有匹配的记录,则不允许对父表对应候选键进行update/delete操作
No action方式
# 同no action, 都是立即检查外键约束
Restrict方式
# 父表有变更时,子表将外键列设置成一个默认的值 但Innodb不能识别
Set default方式
# 写入一行数据
insert into t1 values(1, 'henry', 19);
insert into t1 value(1, 'henry', 19);
# 写入多行数据
insert into t1 values(1, 'henry', 19), (2, 'echo', 18);
# 指定字段写入
insert into t1(name, age) value('henry', 19);
# 删除条件匹配到的数据
delete form 表 where 条件;
# 修改表中数据, set 后的字段可以为一个或多个
update 表 set 字段=值 where 条件;
# 注意null只能使用 is 匹配
where name is null;
SELECT DISTINCT 字段1,字段2... FROM 表名
WHERE 条件
GROUP BY field
HAVING 筛选
ORDER BY field
LIMIT 限制条数
# 查看表中所有数据
select * from 表
# 查看指定字段
select 字段1,字段2... from 表
# 查看指定字段,自动去重
select distinct 字段1,字段2... from 表
# 数值型四则运算,并名别名显示
select name,salary*12 (as) annual_salary form 表
# 数值型四则运算,并名别名, 拼接显示
select concat ('姓名:',name,'薪资:',salary*12) (as) annual_salary form 表
# 使用':'进行拼接
select concat_ws (':', name,salary*12 (as) annual_salary) form 表
# 结合CASE语句:
SELECT(CASE
WHEN emp_name = 'jingliyang' THEN
emp_name
WHEN emp_name = 'alex' THEN
CONCAT(emp_name,'_BIGSB')
ELSE
concat(emp_name, 'SB')
END
) as new_name FROM employee;
select * from t1 where salary>1000;
# 和数值类型无关
select * from t1 where salary=20000 or salary=30000;
# 逻辑运算
select * from t1 where gender='male' and age=18;
# 多选一,可以使用 in
select 字段名,... from t1 where salary in (20000, 30000, 19000);
# not in
select 字段名,... from t1 where salary not in (20000, 30000, 19000);
# is is not
select 字段名 from t1 where 字段 is null;
# between ... and ...
select name,salary from t1 where salary between 10000 and 20000
# like , % 通配符,匹配任意长度,任意内容
select * from t1 where name like '程%';
# like , _ 通配符,匹配一个任意字符
select * from t1 where name like '程_';
# like , 以 n 结尾
select * from t1 where name like '%n';
select * from t1 where name regexp 正则表达式;
SELECT * FROM employee WHERE emp_name REGEXP 'on$';
# 显示一个组的第一个,必须有group的字段
select post from employee group by post;
# distinct 基于group by完成
select count(*/ 主键) from employee;
# 只算id不为空的字段个数
select count(id) from employee;
select avg(salary) from employee;
select sum(salary) from employee;
# 分别对各个组,统计人数
select post,count(*) from employee group by post;
# 对某一组进行统计人数
select post,count(*) from employee where post='teacher';
# 各部门的平均薪资
select post,avg(salary) from employee group by post;
# 最晚入职
select max(hire_date) from employee;
# 最早入职
select min(hire_date) from employee group by post;
# 查询岗位名以及岗位包含的所有员工名字
select post, group_concat(emp_name) from employee group by post;
# 查询岗位名以及各岗位内包含的员工个数
select post, count(id) from employee group by post;
# 查询公司内男员工和女员工的个数
select sex, count(id) from employee group by sex;
# 部门人数大于3
select post from employee group by post having count(*) > 3;
# 平均薪资大于10000
select post from employee group by post having avg(salary) > 10000;
# 过滤整张表,必须有 age 字段,否则报错
select emp_name, age from employee having avg(age)>18;
# 查询各岗位内包含的员工个数小于2的岗位名、岗位内包含员工名字、个数
select post, group_concat(emp_name), count(*) from employee group by post having count(id) < 2;
# 查询各岗位平均薪资大于10000的岗位名、平均工资
select post,avg(salary) from employee group by post having avg(salary) > 10000;
# 查询各岗位平均薪资大于10000且小于20000的岗位名、平均工资
select post,avg(salary) from employee group by post having avg(salary) > 10000 and avg(salary) < 20000;
# 使用 between ... and...
select post,avg(salary) from employee group by post having avg(salary) between 10000 and 20000;
# desc 表示降序排
# asc 表示生序排列,默认值
select * from employee order by salary desc;
# 多个个字段排序,先根据第一个字段排列后,再根据第二个字段排列
select * from employee order by age asc, salary desc;
# 查询各岗位平均薪资大于10000的岗位名、平均工资,结果按平均薪资升序排列
select post, avg(salary) from employee group by post having avg(salary) > 10000 order by avg(salary) asc;
# 查询各岗位平均薪资大于10000的岗位名、平均工资,结果按平均薪资降序排列
select post, avg(salary) from employee group by post having avg(salary) > 10000 order by avg(salary) desc;
# 分页显示
# 默认从0开始,显示前5个
select * from employee limit 5;
# 显示下5个, 5+1 位置开始取
# limit m,n 表示从m+1开始取到5个
limit n offset m等价于limit m,n
select * from employee limit 5,5;
# 显示下5个
select * from employee limit 10,5;
#建表
create table department(id int,
name varchar(20) );
create table staff(id int primary key auto_increment,
name varchar(20),
gender enum('male','female') not null default 'male',
age int,
dep_id int);
#插入数据
insert into department values(200,'技术'),(201,'人力资源'),(202,'销售'),(203,'运营');
insert into staff(name,gender,age,dep_id) values
('henry','male',18,200),
('echo','female',48,201),
('dean','male',38,201),
('diane','female',28,202),
('oleg','male',18,200),
('iris','female',18,204);
#查看表结构和数据
mysql> desc department;
mysql> desc employee;
mysql> select * from department;
mysql> select * from employee;
select 字段列表
FROM 表1 INNER|LEFT|RIGHT JOIN 表2
ON 表1.字段 = 表2.字段;
# 笛卡尔积
select 字段 from t1,t2 where 字段1=字段2;
# 连表查询,staff,department 两个表,和 inner join 效果一致
select * from staff, department as dept where dep_id=dept.id;
select 字段 from t1 inner join t2 on t1(字段1) = t2(字段2);
# t1连接t2,显示全量的左表,只显示匹配到的t2
select 字段 from t1 left join t2 on t1(字段1) = t2(字段2);
select 字段 from t1 right join t2 on t1(字段1) = t2(字段2);
# 全连接
select 字段 from t1 left join t2 on t1(字段1) = t2(字段2) union
select 字段 from t1 right join t2 on t1(字段1) = t2(字段2);
# 通过左外、和右外连接实现全外连接示例
select * from staff left join department as dept on dep_id = dept.id union select * from staff right join department as dept on dep_id = dept.id;
# 以内连接的方式查询staff和department表,并且staff表中的age字段值必须大于25,即找出年龄大于25岁的员工以及员工所在的部门
# 此时括号可以省略
select staff.name, dept.name from staff left join department as dept on dep_id = dept.id where age > 25;
# 以内连接的方式查询staff和department表,并且以age字段的升序方式显示
select * from staff inner join department as dept on dep_id = dept.id order by age;
1:子查询是将一个查询语句嵌套在另一个查询语句中。
2:内层查询语句的查询结果,可以为外层查询语句提供查询条件。
3:子查询中可以包含:IN、NOT IN、ANY、ALL、EXISTS 和 NOT EXISTS等关键字
4:还可以包含比较运算符:= 、 !=、> 、<等
# 子表中匹配到唯一值
select name from emp where dep_id = (select id from department where name='技术');
# 子表中匹配到多个值
select name from emp where dep_id in (select id from department where name in ('技术', '销售'));
# 查询平均年龄在25岁以上的部门名
select name from department where id in (select dep_id from staff group by dep_id having avg(age) > 25);
# 查看技术部员工姓名
select name from staff where dep_id in (select id from department where name = '技术' )
# 查看不足1人的部门名(子查询得到的是有人的部门id)
select name from department where id not in (select dep_id from staff group by dep_id);
# 查询大于所有人平均年龄的员工名与年龄
select name, age from staff where age > (select avg(age) from staff);
# 查询大于部门内平均年龄的员工名、年龄
select name, age from staff t1 inner join (select dep_id, avg(age) avg_age from staff group by dep_id) t2 on t1.dep_id = t2.dep_id where t1.age > t2.avg_age;
# exists后为真
select * from staff where exists (select id from department where id=200);
# 输出staff中所有数据
# exists后为假
select * from staff where exists (select id from department where id=200);
# 输出为空
# 连表查询
select t1.emp_name, t1.hire_date, t1.post from employee as t1 inner join (select depart_id, max(hire_date) as max_date from employee group by depart_id) as t2 on t1.depart_id = t2.depart_id where t1.hire_date = t2.max_date;
# 子查询
select t3.emp_name,t3.post,t3.hire_date from employee as t3 where id in (select (select id from employee as t2 where t2.depart_id=t1.depart_id order by hire_date desc limit 1) from employee as t1 group by depart_id);
# 1.准备表
create table s_test(id int,
name varchar(20),
gender char(6),
email varchar(50));
# 2.创建存储过程,实现批量插入记录
delimiter $$ #声明存储过程的结束符号为$$
create procedure auto_insert()
BEGIN
declare i int default 1;
while(i<3000000)do
insert into s_test values(i,'henry','male',concat('henry',i,'@qq.com'));
set i=i+1;
end while;
END$$ #$$结束
delimiter ; #重新声明分号为结束符号
# 3.查看存储过程
show create procedure auto_insert\G
# 4.调用存储过程
call auto_insert();
# 在写入的时候不更新索引表,只针对myisam生效
ALTER TABLE table_name DELAY_KEY_WRITE= 1;
平衡树 balance tree b树
写入数据:速度较慢,需要整理数据
b树在范围查询b树不占优势(root、leaf、branch)演变成双向链式结构
在b树基础上的改良:b+树(innodb 默认结构)(2)
索引字段要尽量的小:通过上面的分析,我们知道IO次数取决于b+数的高度h,假设当前数据表的数据为N,每个磁盘块的数据项的数量是m,则有h=㏒(m+1)N,当数据量N一定的情况下,m越大,h越小;而m = 磁盘块的大小 / 数据项的大小,磁盘块的大小也就是一个数据页的大小,是固定的,如果数据项占的空间越小,数据项的数量越多,树的高度越低。这就是为什么每个数据项,即索引字段要尽量的小,比如int占4字节,要比bigint8字节少一半。这也是为什么b+树要求把真实的数据放到叶子节点而不是内层节点,一旦放到内层节点,磁盘块的数据项会大幅度下降,导致树增高。当数据项等于1时将会退化成线性表。
树的高度会影响索引的效率
索引特点
mysql中所有的b+树索引的高度都基本上控制在3层
数据库中的B+树索引可以分为聚集索引(clustered index)和辅助索引(secondary index)
聚集索引:数据直接存储在树结构的叶子节点
# InnoDB存储引擎表是索引组织表,即表中数据按照主键顺序存放。
1. 而聚集索引(clustered index)就是按照每张表的主键构造一棵B+树,同时叶子结点存放的即为整张表的行记录数据,也将聚集索引的叶子结点称为数据页。
2. 聚集索引的这个特性决定了索引组织表中数据也是索引的一部分。同B+树数据结构一样,每个数据页都通过一个双向链表来进行链接。
# 如果未定义主键,MySQL取第一个唯一索引(unique)而且只含非空列(NOT NULL)作为主键,InnoDB使用它作为聚簇索引。
1. 如果没有这样的列,InnoDB就自己产生一个这样的ID值,它有六个字节,而且是隐藏的,使其作为聚集索引。
# 由于实际的数据页只能按照一棵B+树进行排序,因此每张表只能拥有一个聚集索引。
1. 在多数情况下,查询优化器倾向于采用聚集索引。因为聚集索引能够在B+树索引的叶子节点上直接找到数据。
2. 此外由于定义了数据的逻辑顺序,聚集索引能够特别快地访问针对范围值得查询。
alter table t1 add primary key(id);
alter table t1 modify id not null unique;
辅助索引:数据不直接存储在树中