本篇主涉及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层级优化的更多相关文章

  1. MySQL 数据库--SQL语句优化

    explain查询和分析sql 开发中,为满足一业务功能,使用mysql书写sql时,一条sql往往有多种写法,那么我们就需要选择执行效率比较高的sql. 因此要比较分析sql的执行过程,且同一条sq ...

  2. 千万级大数据的Mysql数据库SQL语句优化

    1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索 ...

  3. 我的mysql数据库sql优化原则

    原文 我的mysql数据库sql优化原则 一.前提 这里的原则 只是针对mysql数据库,其他的数据库 某些是殊途同归,某些还是存在差异.我总结的也是mysql普遍的规则,对于某些特殊情况得特殊对待. ...

  4. MySQL性能调优与架构设计——第8章 MySQL数据库Query的优化

    第8章 MySQL数据库Query的优化 前言: 在之前“影响 MySQL 应用系统性能的相关因素”一章中我们就已经分析过了Query语句对数据库性能的影响非常大,所以本章将专门针对 MySQL 的 ...

  5. php面试专题---MySQL常用SQL语句优化

    php面试专题---MySQL常用SQL语句优化 一.总结 一句话总结: 原理,万变不离其宗:其实SQL语句优化的过程中,无非就是对mysql的执行计划理解,以及B+树索引的理解,其实只要我们理解执行 ...

  6. 高并发大流量专题---10、MySQL数据库层的优化

    高并发大流量专题---10.MySQL数据库层的优化 一.总结 一句话总结: mysql先考虑做分布式缓存,过了缓存后就做mysql数据库层面的优化 1.mysql数据库层的优化的前面一层是什么? 数 ...

  7. MySQL 数据库SQL语句——高阶版本2

    MySQL 数据库SQL语句--高阶版本2 实验准备 数据库表配置: mysql -uroot -p show databases; create database train_ticket; use ...

  8. 第二百八十一节,MySQL数据库-SQL注入和pymysql模块防止SQL注入

    MySQL数据库-SQL注入和pymysql模块防止SQL注入 SQL注入就是通过SQL语句绕开程序判断,获取到数据库的内容 下面以一个简单的程序登录SQL注入举例: 正常登录 1.数据库有一张会员表 ...

  9. mysql的sql语句优化方法面试题总结

    mysql的sql语句优化方法面试题总结 不要写一些没有意义的查询,如需要生成一个空表结构: select col1,col2 into #t from t where 1=0 这类代码不会返回任何结 ...

随机推荐

  1. 三个重要的游标sp_cursoropen

    請問這三個存諸過程的作用是什么﹖ sp_cursoropen, sp_cursorfetch, sp_cursorclose API 服务器游标实现  SQL Server OLE DB 提供程序. ...

  2. Hbase常见异常

    1. HBase is able to connect to ZooKeeper but the connection closes immediately hbase(main):001:0> ...

  3. JSP EL

    一.JSP EL语言定义 E L(Expression Language)  目的:为了使JSP写起来更加简单. 表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 ...

  4. python安装——Windows平台

    刚刚申请博客,第一次写随笔,记录下自己最近的学习情况,希望自己能够不断的学习,不断的丰富自己~ 最近刚开始学python,记录一下,希望大家相互学习,批评指正~ 1.下载python:https:// ...

  5. Scala学习笔记--隐式转换

    隐式转换的规则:1.无歧义规则:隐式转换唯有不存在其他可插入转换的前提下才能插入  若编译器有两种方法修正x+y 如convert1(x)+y,convert2(x)+y,会报错2.单一调用规则:只尝 ...

  6. LDD命令--可执行文件依赖的库出现错误时

    http://littlepig3056.blog.163.com/blog/static/180758353201212751814134/ ldd  查看可执行文件依赖的库,结果会列出依赖的库名及 ...

  7. 《Programming WPF》翻译 第7章 5.可视化层编程

    原文:<Programming WPF>翻译 第7章 5.可视化层编程 形状元素能提供一种便利的方式与图形一起工作,在一些情形中,添加表示绘图的元素到UI树中,可能是比它的价值更加麻烦.你 ...

  8. hihoCoder 1098

    题目大意:n 个城市由 m 条边连接,每条边有权值,求将所有城市连接起来时的最小权值和. 代码: #include<iostream> #include<cstdio> #in ...

  9. 【POJ2739】Sum of Consecutive Prime Numbers

    简单的素数打表,然后枚举.开始没注意n读到0结束,TLE了回..下次再认真点.A过后讨论里面有个暴力打表过的,给跪了! #include <iostream> #include <c ...

  10. 案例:用JS实现放大镜特效

    案例:用JS实现放大镜特效 案例:用JS实现放大镜特效