MySQL数据库SQL层级优化
本篇主涉及MySQL SQL Statements层面的优化。
首先,推荐一个链接为万物之始:http://dev.mysql.com/doc/refman/5.0/en/optimization.html
其次,Explain作为分析SQL的优化利器,SHOW STATUS 和 PROCEDURE ANALYSE(16, 256)也蛮有用。推荐两篇MySQL Explain:
http://www.khankennels.com/presentations/pdf/explain.pdf
http://dev.mysql.com/doc/refman/5.0/en/explain-output.html
1、一次INSERT多条语句
避免循环单条插入,代价很昂贵!在IBATIS中一次插入多条语句配置:
<insert id="insertUserList" parameterClass="java.util.List">
<![CDATA[
insert into user(
id,
userName,
passWord
) values
]]]]>
<iterate conjunction=",">
<![CDATA[
(
#list[].id#,
#list[].userName#,
#list[].passWord#
)
]]]]>
</iterate>
</insert>
2、有效利用索引
-Index Unique Column。在MySQL中使用唯一索引会提升效率,仅当作为Search目的、才有必要设置。
-在WHERE条件中尽量使用索引。
-考虑联合索引,但存在”first hit”问题。
-FORCE INDEX强制使用指定索引列表,SELECT SQL_BUFFER_RESULTS强制使用MySQL生成临时结果集(使得好临时结果集、将大大提升性能,还有SQL_SMALL_RESULT、SQL_BIG_RESULT),USE INDEX给定参考索引列表,IGNORE INDEX给定忽略索引列表。
-避免在索引列使用IS NULL或NOT IS NULL。
3、一定要使用LIMIT 1
大数据集,会占用内存、带宽等资源。使用LIMIT,强迫分页,减少服务器压力。
4、尽可能地使用NOT NULL,无论是在WHERE查询还是表字段设计中使用默认值。
5、Utilize Union instead of OR
Indexes lose their speed advantage when using them in OR-situations in MySQL at least. Hence, this will not be useful although indexes is being applied. 例:
SELECT * FROM EventPrizeUser A WHERE A.`UserID`=39235750 OR A.`UserMobile`='18961751810'
vs.
(SELECT * FROM EventPrizeUser WHERE `UserID`=39235750)
UNION
(SELECT * FROM EventPrizeUser WHERE `UserMobile`='18961751810')
第一条走index_merge,第二条走ref(const)。ref是要优于index_merge,虽然该条语句可能OR的性能略高于UNION(约1ms),但UNION可以保证一定走索引,而MySQL的OR执行计划不走index_merge的概率也蛮高。OR的每个条件列都必须使用索引,OR才使用索引。
6、使用合适确数据类型、缩减存储空间
-使用ENUM、而不是VARCHAR。ENUM利用TINYINT、类型紧凑、比较快,但却可以有字符串的“华丽外表”。如果是预定义好的类型,可以尝试SET类型。?ENUM新增类型。使用PROCEDURE ANALYSE分析出表的ENUM建议。
-使用DATE、TIMESTAMP,避免DATETIME。TIMESTAMP的存储空间是DATETIME的一半。
7、避免不必要排序,如DISTINCT等都会触发排序
-GROUP BY A ORDER BY NULL。GROUP BY默认会使用排序,所以如果结果集比较大、可以采用ORDER BY NULL去掉。
-ORDER BY,仅对WHERE中同个组合索引内的key采用统一ASC/DESC方式
例:SELECT * FROM WHERE part_key1 ORDER BY part_key1 DESC, part_key2 DESC
8、慎用NOT,避免使用IN、 NOT IN、<>、OR或HAVING等
用EXIST、NOT EXISTS代替IN、NOT EXISTS,因为可以直接走关联子句的WHERE。<>可以用 “> & <”代替。如:
SELECT MemberCardID FROM `MC_MemberCard` WHERE MemberCardID <> 1247
vs.
SELECT MemberCardID FROM `MC_MemberCard` WHERE MemberCardID < 1247 OR MemberCardID > 1247
faster 1ms
9、Wildcard,LIKE ‘a%’,NOT ‘%a%’
’a%’为前缀匹配、走索引,但’%a%’导致全表查询。
10、不要以字符形式声明数字
a=1、NOT a = ‘1’,因为会使索引失效、导致全表扫描。?会么?
11、禁用SELECT FOR UPDATE
FOR UPDATE属于悲观锁(Pessimistic Locking),在整个数据处理过程中将处于锁定状态。乐观锁(Optimistic Locking)则采用更加宽松的锁机制。wiki定义如下:
Optimistic concurrency control (OCC) is a concurrency control method for relational database management systems that assumes that multiple transactions can complete without affecting each other, and that therefore transactions can proceed without locking the data resources that they affect. Before committing, each transaction verifies that no other transaction has modified its data. If the check reveals conflicting modifications, the committing transaction rolls back。
乐观锁最常用方式是通过version或TIMESTAMP,防止数据不一致问题。修改数据时可利用行写锁保证唯一性:
UPDATE T SET VERSION+1=VERSION WHERE ID=xxx
Hibernate在框架支持乐观锁机制,IBATIS中暂时没有相应支持,但可参考下文:
http://matejtymes.blogspot.hk/2010/11/optimistic-locking-on-ibatis.html
还可使用前置条件解决并发问题,如:
UPDATE STATUS=’BUY_SUCC’ WHERE OrderId=xxx AND Status=’WAITING_PAY’
12、垂直分割
水平分割、SQL太复杂不好处理,但经验而言,一般情景下是没有太多效率提升,可以将查询频烦、固定表长的部分作为一部分。?啥时候该做这件事?
13、高并发写操作的表,不建议使用自增ID
使用自增ID、会引起写锁保护,也不能使用MySQL的UUID(),因为会导致主备数据不一致。并发应用程序中生成ID,保证唯一性、推荐两种方式:
-经典的combined guid/timestamp方式:占32字节,效率太慢。利用BitConverter.ToInt64()转换成8个字节,可以接受的友好;
-根据业务规则自定义方案。如:12位年月日时分秒+3位服务器编码+3位表编码+5位随机码/流水码。?啥级别自增会防碍读写?
14、使用Prepared Statements(JDBC)
次少SQL解析、生成执行计划次数,顺带过滤注入。在IBATIS中,#{id}表示PreparedStatement parameter,在XML语句配制中有statementType参数,默认为PREPARED。
再送一个SQL优化的网站:
http://www.mysqlperformanceblog.com/
MySQL数据库SQL层级优化的更多相关文章
- MySQL 数据库--SQL语句优化
explain查询和分析sql 开发中,为满足一业务功能,使用mysql书写sql时,一条sql往往有多种写法,那么我们就需要选择执行效率比较高的sql. 因此要比较分析sql的执行过程,且同一条sq ...
- 千万级大数据的Mysql数据库SQL语句优化
1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索 ...
- 我的mysql数据库sql优化原则
原文 我的mysql数据库sql优化原则 一.前提 这里的原则 只是针对mysql数据库,其他的数据库 某些是殊途同归,某些还是存在差异.我总结的也是mysql普遍的规则,对于某些特殊情况得特殊对待. ...
- MySQL性能调优与架构设计——第8章 MySQL数据库Query的优化
第8章 MySQL数据库Query的优化 前言: 在之前“影响 MySQL 应用系统性能的相关因素”一章中我们就已经分析过了Query语句对数据库性能的影响非常大,所以本章将专门针对 MySQL 的 ...
- php面试专题---MySQL常用SQL语句优化
php面试专题---MySQL常用SQL语句优化 一.总结 一句话总结: 原理,万变不离其宗:其实SQL语句优化的过程中,无非就是对mysql的执行计划理解,以及B+树索引的理解,其实只要我们理解执行 ...
- 高并发大流量专题---10、MySQL数据库层的优化
高并发大流量专题---10.MySQL数据库层的优化 一.总结 一句话总结: mysql先考虑做分布式缓存,过了缓存后就做mysql数据库层面的优化 1.mysql数据库层的优化的前面一层是什么? 数 ...
- MySQL 数据库SQL语句——高阶版本2
MySQL 数据库SQL语句--高阶版本2 实验准备 数据库表配置: mysql -uroot -p show databases; create database train_ticket; use ...
- 第二百八十一节,MySQL数据库-SQL注入和pymysql模块防止SQL注入
MySQL数据库-SQL注入和pymysql模块防止SQL注入 SQL注入就是通过SQL语句绕开程序判断,获取到数据库的内容 下面以一个简单的程序登录SQL注入举例: 正常登录 1.数据库有一张会员表 ...
- mysql的sql语句优化方法面试题总结
mysql的sql语句优化方法面试题总结 不要写一些没有意义的查询,如需要生成一个空表结构: select col1,col2 into #t from t where 1=0 这类代码不会返回任何结 ...
随机推荐
- Java 中关键字transient引出序列化与反序列化
一:transient(临时的)关键字 1.transient关键字只能修饰变量,而不能修饰方法和类.注意,本地变量是不能被transient关键字修饰的. 2.被transient关键字修饰的变量不 ...
- 记一道css面试题 : 三栏布局两边宽度固定,中间宽度自适应,并且布局随屏幕大小改变。
前几天面试时有道css题没做出来,回来好好学习一番后把其记录下来. 题目是这样的:左中右三栏布局,左右两栏宽度固定,左右两栏的宽度为200像素,中间栏宽度自适应.当屏幕小于600px时,3栏会分别占用 ...
- TypeScript开发Vue
用TypeScript开发Vue——如何通过vue实例化对象访问实际ViewModel对象 目录 背景 解决方案 关于Vue中的计算属性类型 TypeScript的强制类型声明语法 强制类型声明的局限 ...
- Scala学习笔记--抽象成员
package com.evor.test1 class Test1 { } object Test1{ def main(args:Array[String]):Unit = { //类参数和抽象字 ...
- 2015北大夏令营day1 B:An Idea of Mr. A
题意:给定一个范围l,r计算i,j(i<j)属于这个范围内的gcd(2^(2^i)+1,2^(2^j)+1)的总和. 思路:费马数的应用,让我惊呆的是当年居然有123个人会做,我tm毛都不会.. ...
- poj2409 Let it Bead
Let it Bead Time Limit: 1000MS M ...
- Windows 下 pip和easy_install 的安装与使用
最简单的莫过于用vs 开发python 的程序,集成PTVS插件后,果然简单易用,调试也方便. 手工搭建的话: 官网:https://pypi.python.org/pypi/setuptools 主 ...
- PS抠图神器: KnockOut 2.0安装汉化和使用教程
PS抠图神器: KnockOut 2.0安装汉化和使用教程 http://jingyan.baidu.com/article/6b97984d8aeadc1ca2b0bf3b.html
- 【转】Linux系统性能分析命令
作为一名linux系统管理员,最主要的工作是优化系统配置,使应用在系统上以最优的状态运行,但是由于硬件问题.软件问题.网络环境等的复杂性和多变性,导致对系统的优化变得异常复杂,如何定位性能问题出在哪个 ...
- C# 二叉堆
二叉堆数据结构讲解: http://www.cnblogs.com/yc_sunniwell/archive/2010/06/28/1766751.html C#代码实现 using System ...