来源:juejin.im/post/59b11ba151882538cb1ecbd0mysql
本文主要针对的是关系型数据数据库MySql。键值类数据库能够参考:sql
先简单梳理下Mysql的基本概念,而后分建立时和查询时这两个阶段的优化展开。json
1.1 逻辑架构缓存
第一层:客户端经过链接服务,将要执行的sql指令传输过来安全
第二层:服务器解析并优化sql,生成最终的执行计划并执行性能优化
1.2 锁服务器
数据库经过锁机制来解决并发场景-共享锁(读锁)和排他锁(写锁)。读锁是不阻塞的,多个客户端能够在同一时刻读取同一个资源。写锁是排他的,而且会阻塞其余的读锁和写锁。简单提下乐观锁和悲观锁。数据结构
乐观锁,一般用于数据竞争不激烈的场景,多读少写,经过版本号和时间戳实现。架构
要锁定数据须要必定的锁策略来配合。
表锁,锁定整张表,开销最小,可是会加重锁竞争。
可是MySql的存储引擎的真实实现不是简单的行级锁,通常都是实现了多版本并发控制(MVCC)。MVCC是行级锁的变种,多数状况下避免了加锁操做,开销更低。MVCC是经过保存数据的某个时间点快照实现的。
1.3 事务
事务保证一组原子性的操做,要么所有成功,要么所有失败。一旦失败,回滚以前的全部操做。MySql采用自动提交,若是不是显式的开启一个事务,则每一个查询都做为一个事务。
隔离级别控制了一个事务中的修改,哪些在事务内和事务间是可见的。四种常见的隔离级别:
未提交读(Read UnCommitted),事务中的修改,即便没提交对其余事务也是可见的。事务可能读取未提交的数据,形成脏读。
提交读(Read Committed),一个事务开始时,只能看见已提交的事务所作的修改。事务未提交以前,所作的修改对其余事务是不可见的。也叫不可重复读,同一个事务屡次读取一样记录可能不一样。
可重复读(RepeatTable Read),同一个事务中屡次读取一样的记录结果时结果相同。
1.4 存储引擎
InnoDB引擎,最重要,使用最普遍的存储引擎。被用来设计处理大量短时间事务,具备高性能和自动崩溃恢复的特性。
MyISAM引擎,不支持事务和行级锁,崩溃后没法安全恢复。
2.1 Schema和数据类型优化
整数
TinyInt,SmallInt,MediumInt,Int,BigInt 使用的存储8,16,24,32,64位存储空间。使用Unsigned表示不容许负数,可使正数的上线提升一倍。
实数
Float,Double , 支持近似的浮点运算。
字符串
VarChar,存储变长的字符串。须要1或2个额外的字节记录字符串的长度。
Char,定长,适合存储固定长度的字符串,如MD5值。
时间类型
DateTime,保存大范围的值,占8个字节。
优化建议点
尽可能使用对应的数据类型。好比,不要用字符串类型保存时间,用整型保存IP。
选择更小的数据类型。能用TinyInt不用Int。
标识列(identifier column),建议使用整型,不推荐字符串类型,占用更多空间,并且计算速度比整型慢。
不推荐ORM系统自动生成的Schema,一般具备不注重数据类型,使用很大的VarChar类型,索引利用不合理等问题。
真实场景混用范式和反范式。冗余高查询效率高,插入更新效率低;冗余低插入更新效率高,查询效率低。
建立彻底的独立的汇总表\缓存表,定时生成数据,用于用户耗时时间长的操做。对于精确度要求高的汇总操做,能够采用 历史结果+最新记录的结果 来达到快速查询的目的。
2.2 索引
索引包含一个或多个列的值。MySql只能高效的利用索引的最左前缀列。索引的优点:
减小查询扫描的数据量
避免排序和零时表
B-Tree
使用最多的索引类型。采用B-Tree数据结构来存储数据(每一个叶子节点都包含指向下一个叶子节点的指针,从而方便叶子节点的遍历)。B-Tree索引适用于全键值,键值范围,键前缀查找,支持排序。
B-Tree索引限制:
若是不是按照索引的最左列开始查询,则没法使用索引。
不能跳过索引中的列。若是使用第一列和第三列索引,则只能使用第一列索引。
哈希索引
只有精确匹配索引的全部列,查询才有效。存储引擎会对全部的索引列计算一个哈希码,哈希索引将全部的哈希码存储在索引中,并保存指向每一个数据行的指针。
哈希索引限制:
没法用于排序
不支持部分匹配
优化建议点
注意每种索引的适用范围和适用限制。
索引的列若是是表达式的一部分或者是函数的参数,则失效。
针对特别长的字符串,可使用前缀索引,根据索引的选择性选择合适的前缀长度。
使用多列索引的时候,能够经过 AND 和 OR 语法链接。
重复索引不必,如(A,B)和(A)重复。
索引在where条件查询和group by语法查询的时候特别有效。
将范围查询放在条件查询的最后,防止范围查询致使的右边索引失效的问题。
3.1 查询质量的三个重要指标
响应时间 (服务时间,排队时间)
扫描的行
3.2 查询优化点
避免查询无关的列,如使用Select * 返回全部的列。
避免查询无关的行
切分查询。将一个对服务器压力较大的任务,分解到一个较长的时间中,并分屡次执行。如要删除一万条数据,能够分10次执行,每次执行完成后暂停一段时间,再继续执行。过程当中能够释放服务器资源给其余任务。
分解关联查询。将多表关联查询的一次查询,分解成对单表的屡次查询。能够减小锁竞争,查询自己的查询效率也比较高。由于MySql的链接和断开都是轻量级的操做,不会因为查询拆分为屡次,形成效率问题。
注意count的操做只能统计不为null的列,因此统计总的行数使用count(*)。
group by 按照标识列分组效率高,分组结果不宜出行分组列以外的列。
关联查询延迟关联,能够根据查询条件先缩小各自要查询的范围,再关联。
SELECT id, NAME, age WHERE student s1 INNER JOIN ( SELECT id FROM student ORDER BY age LIMIT 50,5 ) AS s2 ON s1.id = s2.id
Union查询默认去重,若是不是业务必须,建议使用效率更高的Union All
来自大神-小宝
1.条件中的字段类型和表结构类型不一致,mysql会自动加转换函数,致使索引做为函数中的参数失效。
2.like查询前面部分未输入,以%开头没法命中索引。
3.补充2个5.7版本的新特性:
generated column,就是数据库中这一列由其余列计算而得
CREATE TABLE triangle (sidea DOUBLE, sideb DOUBLE, area DOUBLE AS (sidea * sideb / 2)); insert into triangle(sidea, sideb) values(3, 4); select * from triangle;
+-------+-------+------+ | sidea | sideb | area | +-------+-------+------+ | 3 | 4 | 6 | +-------+-------+------+
支持JSON格式数据,并提供相关内置函数
CREATE TABLE json_test (name JSON); INSERT INTO json_test VALUES('{"name1": "value1", "name2": "value2"}'); SELECT * FROM json_test WHERE JSON_CONTAINS(name, '$.name1');
来自JVM专家-达
关注explain在性能分析中的使用
EXPLAIN SELECT settleId FROM Settle WHERE settleId = "3679"
select_type,有几种值:simple(表示简单的select,没有union和子查询),primary(有子查询,最外面的select查询就是primary),union(union中的第二个或随后的select查询,不依赖外部查询结果),dependent union(union中的第二个或随后的select查询,依赖外部查询结果)
type,有几种值:system(表仅有一行(=系统表),这是const链接类型的一个特例),const(常量查询), ref(非惟一索引访问,只有普通索引),eq_ref(使用惟一索引或组件查询),all(全表查询),index(根据索引查询全表),range(范围查询)
possible_keys: 表中可能帮助查询的索引
key,选择使用的索引
key_len,使用的索引长度
rows,扫描的行数,越大越很差
欢迎你们关注个人公众号【风平浪静如码】,海量Java相关文章,学习资料都会在里面更新,整理的资料也会放在里面。
以为写的还不错的就点个赞,加个关注呗!点关注,不迷路,持续更新!!!