当前位置:Gxlcms > 数据库问题 > MySQL

MySQL

时间:2021-07-01 10:21:17 帮助过:18人阅读

一 事务

事务:指逻辑上的一组操作,组成这组操作的各个逻辑单元,要么全都成功,要么全都失败。

事务的特性:

原子性:事务的不可分割,组成事务的各个逻辑单元不可分割。

一致性:事务执行的前后,数据完整性保持一致。

隔离性:事务执行不应该受到其他事务的干扰。

持久性:事务一旦结束,数据就持久化到数据库中。

如果不考虑隔离性,会引发一些安全问题,主要体现在读取数据上:

脏读:   一个事务读到了另一个事务未提交的数据,导致查询结果不一致

不可重复读:一个事务读到了另一个事务已经提交的update的数据,导致多次查询结果不一致。

虚读/幻读:  一个事务读到了另一个事务已经提交的insert的数据,导致多次查询结果不一致。

设置事务的隔离级别:

read uncommitted:脏读,不可重复读,虚读都有可能发生

read committed :  避免脏读。但是不可重复读和虚读是有可能发生

repeatable read :  避免脏读和不可重复读,但是虚读有可能发生

serializable :    避免脏读,不可重复读,虚读

设置事务隔离级别:SET SESSION TRANSACTION ISOLATION LEVEL 隔离级别

查看当前隔离级别:SELECT @@tx_isolation;

二 常用SQL语句

结构化查询语言(Structured Query Language)简称SQL

建库:

create database 数据库名;

create database 数据库名 character set 字符集;


查看所有库: show databases;

查看库定义信息: show create database 数据库名;

删库: drop database 数据库名称;

切换库: use 数据库名;

查看正在使用的数据库: select database();

建表:

create table 表名(
字段名 类型(长度) 约束,
字段名 类型(长度) 约束
);

查看所有表:show tables;

查看表结构:desc 表名;

删表:drop table 表名;


约束

A: 约束的作用:限制加入表的数据类型,限制每一列能写什么数据,不能写什么数据。用来保证数据完整性(多表之间)

create table 表名(
列名 类型(长度) 约束,
列名 类型(长度) 约束
);

B: 哪些约束:

主键约束

非空约束

唯一约束

外键约束

主键约束

* A:用于标识当前记录的字段。它的特点是非空,唯一。一般情况下主键不具备任何含义,只用于标识当前记录。

* B: 格式:

1.在创建表时创建主键,在字段后面加上 primary key.

create table tablename(
id int primary key,
.......
)

2. 在创建表时创建主键,在表创建的最后来指定主键

create table tablename(
id int,
.......,
primary key(id)
)

3.删除主键:alter table 表名 drop primary key;

4.主键自动增长:一般主键是自增长的字段,不需要指定。实现添加自增长语句,主键字段后加auto_increment(只适用MySQL)

例如:

CREATE TABLE sort (
sid INT PRIMARY KEY auto_increment, #分类ID
sname VARCHAR(100) #分类名称
);

常见表的操作

修改表结构

* A: 添加列

alter table 表名 add 列名 类型(长度) 约束;

* B: 修改列的类型长度及约束

alter table 表名 modify 列名 类型(长度) 约束;

* C: 修改列名

alter table 表名 change 旧列名 新列名 类型(长度) 约束;

* D: 删除列

alter table 表名 drop 列名;

* E: 修改表名

rename table 表名 to 新表名;

* F: 修改表的字符集

salter table 表名 character set 字符集;

数据表添加数据_1

insert into 表 (列名1,列名2,列名3..) values (值1,值2,值3..); -- 向表中插入某些列

数据表添加数据_2

* A: 添加数据格式,不考虑主键

insert into 表名 (列名) values (值)

* B: 添加数据格式,所有值全给出

insert into 表名 values (值1,值2,值3..); --向表中插入所有列

* C: 添加数据格式,批量写入

insert into 表名 (列名1,列名2,列名3) values (值1,值2,值3),(值1,值2,值3)


更新数据

update 表名 set 字段名=值,字段名=值;

update 表名 set 字段名=值,字段名=值 where 条件;

* B: 注意:值如果是字符串或者日期需要加’’.


删除数据

delete from 表名 where 条件;

或者

truncate table 表名;

区别:

delete 一条一条删除,不清空auto_increment记录数。

truncate 直接删表并重新建表,auto_increment将置为零,从新开始。


* C: distinct用于去除重复记录

select distinct 字段 from 表名;

例如:

select distinct money from zhangwu;

sql语句的操作可以直接对列进行运算。

例如:将所有账务的金额+10000元进行显示.

select pname,price+10000 from product;


LIKE 通配符 -----------模糊查询,Like语句中有两个通配符:

% 用来匹配多个字符;例first_name like ‘a%’;

_ 用来匹配一个字符。例first_name like ‘a_’;

not ------------ 不成立,例:where not(salary>100);


聚合函数

* A: 聚合函数

* 之前我们做的查询都是横向查询,它们都是根据条件一行一行的进行判断,聚合函数查询是纵向查询,它是对一列的值进行计算,然后返回一个单一的值;聚合函数会忽略空值。

SELECT COUNT(*)AS‘count‘ FROM zhangwu

SELECT SUM(zmoney) FROM zhangwu

SELECT SUM(zname) FROM zhangwu WHERE zname LIKE‘%收入%‘

SELECT MAX(zmoney) FROM zhangwu

SELECT AVG(zmoney)FROM zhangwu


分组查询

* 分组查询是指使用group by字句对查询信息进行分组,例如:我们要统计出zhanguw表中所有分类账务的总数量,这时就需要使用group by 来对zhangwu表中的账务信息根据parent进行分组操作。

* SELECT 字段1,字段2… FROM 表名 GROUP BY 字段 HAVING 条件;

* 分组操作中的having子语句,是用于在分组后对数据进行过滤的,作用类似于where条件。

* b: having与where的区别

* having是在分组后对数据进行过滤.

* where是在分组前对数据进行过滤

* having后面可以使用分组函数(统计函数)

* where后面不可以使用分组函数。


* B: 案例代码

分组查询: group by 被分组的列名

必须跟随聚合函数

select 查询的时候,被分组的列,要出现在select 选择列的后面

SELECT SUM(zmoney),zname FROM zhangwu GROUP BY zname

-- 对zname内容进行分组查询求和,但是只要支出

SELECT SUM(zmoney)AS ‘getsum‘,zname FROM zhangwu WHERE zname LIKE‘%支出%‘

GROUP BY zname

ORDER BY getsum DESC

-- 对zname内容进行分组查询求和,但是只要支出, 显示金额大于5000

-- 结果集是分组查询后,再次进行筛选,不能使用where, 分组后再次过滤,关键字 having

SELECT SUM(zmoney)AS ‘getsum‘,zname FROM zhangwu WHERE zname LIKE‘%支出%‘

GROUP BY zname HAVING getsum>5000

在员工表上添加外键

alter table employee add foreign key (dno) references dept(did);

设置外键为非空

alter table employee modify dno int not null;

查询班级名称,和班级总人数

SELECT c.cname,COUNT(*) FROM classes c,student s

WHERE c.cid = s.cno

GROUP BY c.cname;

查询学生的姓名和学生所选的总课程平均成绩。

select s.sname,avg(sc.score) from student s,stu_cour sc

where s.sid = sc.sno

group by s.sname;

查询学生的姓名和学生的选课总数,显示选课超过2门学生姓名。

select s.sname,count(*) from student s,stu_cour sc

where s.sid = sc.sno

group by s.sname

having count(*) > 2;

查询平均成绩大于80分的学生的总数。

select count(*) from student s

where s.sid in (SELECT sc.sno FROM stu_cour sc GROUP BY sc.sno HAVING AVG(sc.score) > 80);

查询学生和平均成绩,但是平均成绩大于01班的任何一个学生的评价成绩。

SELECT s.sname,AVG(sc.score) FROM student s,stu_cour sc

WHERE s.sid = sc.sno

GROUP BY s.sname HAVING AVG(sc.score) > ANY(SELECT AVG(sc.score) FROM student s,stu_cour sc,classes c WHERE s.sid = sc.sno AND s.cno = c.cname AND c.cname =‘01班‘ GROUP BY s.sname);

 三 SQL优化

1. 添加索引


1). 什么时候考虑添加索引?  数据量大; 查询频率高;


2). 针对什么样的字段添加索引?  经常在查询条件中出现的字段; 数据波动范围大字段;


3). 数据库表的索引太多会有什么影响?  可能影响插入的性能;


4). 索引的使用规则?

explain 解释器来查看select语句的执行计划 ;

比如 : explain select * from tbl_teacher where name = "zhangsan";

2. 尽量不适用 select * , 使用具体字段代替

3. where 和 having, where子句中不能跟组函数. 当使用where 和 having 都可以使用时, 尽量使用where.

1). select deptno, avg(sal) from emp where deptno = 10 group by deptno; ---> 先过滤,后分组, 分组的对象数据少, 效率高.

2). select deptno, avg(sal) from emp group by deptno having deptno = 10; ---> 先分组后过滤.

4. 如果需要查询多张表, 可以使用子查询, 也可以使用多表查询时, 理论上,尽量使用多表查询 ---> 因为性能高.

因为子查询, 对数据操作两次或多次, 而多表查询则只操作一次, 所以效率高---------> 不考虑笛卡尔积的情况下.

但是,如果表中的数据太多, 多表产生的笛卡尔积太大, 会影响多表查询的效率.

5. 尽量不使用集合运算.

参与运算的集合越多, 运行效率越低. UNION(并集) , UNION ALL(全并集) , INTERCEPT(交集) , MINUS(差集)

6. 尽量不适用in操作符.

ORACLE试图将其转换成多个表的连接,如果转换不成功则先执行IN里面的子查询,再查询外层的表记录,如果转换成功则直接采

用多个表的连接方式查询。由此可见用IN的SQL至少多了一个转换的过程

在业务密集的SQL当中尽量不采用IN操作符,用EXISTS 方案代替。

7. 强烈不建议使用 not in 操作符.因为它不能应用表的索引; 可以使用not exists来替代.

8. IS NULL 或IS NOT NULL操作(判断字段是否为空),判空一般是不会应用索引的,因为索引不索引空值;

  用其它相同功能的操作运算代替,如:a is not null 改为 a>0 或a>’’等。不允许字段为空,

9. LIKE的模糊查询如果用的不好也会降低性能,如LIKE ‘%5400%‘ 这种查询不会引用索引,而LIKE ‘5400%‘则会使用索引

10. 查询表顺序的影响

 

MySQL

标签:cno   ESS   标识   功能   需要   ali   理论   serial   dep   

人气教程排行