栏目分类:
子分类:
返回
终身学习网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
终身学习网 > IT > 软件开发 > 后端开发 > PHP > php开源框架 > xunsearch

万字MySQL调优,从sql到索引到配置文件详解(精)EXPLAIN详解

xunsearch 更新时间:发布时间: 百科书网 趣学号

MySQL调优
      • 三步骤
  • 一、MySQL逻辑架构
  • 二、SQL如何执行
    • 2-1、查询语句:
    • 2-2、更新语句:
    • 2-3、删除:
      • 数据空洞
        • 其实到这里基本思路已经可以清晰的看到了,接下来就是实际操作来证实一些猜想,可能有遗漏的欢迎评论区补全
  • 三、优化
    • 3-1、数据库表结构的优化
    • 3-2、SQL语句优化
    • 3-3、索引的优化
    • 3-4、系统配置的优化
    • 3-5、硬件的优化

三步骤

1.认识MySQL
2.确定调优方向
3.MySQL调优

一、MySQL逻辑架构

逻辑图:

查询流程:
1.建立TCP连接,进行用户身份认证,拿到用户的权限数据
2.查缓存,命中则直接返回
3.解析器解析SQL,分两步,词法分析(提取查哪个表,哪些字段,有什么where条件),语法分析(主要判断sql是否正确,是否符合mysql语法)
4.优化器,选择它认为最优的执行方案,例如索引的选择,多表查询的关联顺序等等
5.执行器,校验权限,调用引擎的接口返回结果

二、SQL如何执行

InnoDB引擎的MVCC是通过在每行记录后面保存两个隐藏列实现的,分别为创建版本号和过期版本号。
基于这两个隐藏字段,sql可以分为两种,查询(select)更新(增,改,删)。

2-1、查询语句:
SELECT * FROM t_student a WHERe a.age='15' AND a.name='小明';

结合上面的说明,梳理一下该条sql的执行流程:

  • 先检查该语句是否有权限,如果没有权限,直接返回错误信息,如果有权限,在mysql8.0版本以前,会先查询缓存,以这条sql语句为key在内存中查询是否有结果,如果有直接缓存,如果没有,执行下一步。
  • 通过分析器进行词法分析,提取sql语句的关键元素,比如提取上面这个语句是查询select,提取需要查询的表名为tb_student,需要查询所有的列,查询条件是这个表的id=‘1’。然后判断这个sql语句是否有语法错误,比如关键词是否正确等等,如果检查没问题就执行下一步。
  • 优化器进行确定执行方案,根据WHERe条件可以有两种执行方案:
1.查询t_student 表中name为“小明”的记录,然后判断age是否等于15
2.查询t_student 表中age等于15的记录,然后判断name是否为“小明”
  • 优化器会根据优化算法选择执行效率最好的一个方案(虽然可能不一定最好)。
  • 权限校验,没有权限则返回错误信息,否则调用数据库引擎接口返回执行结果。
2-2、更新语句:
UPDATE t_student a SET a.age='18' WHERe a.name='小明';

这条sql要做的事情就是把name为“小明”的记录age字段更新为15,那么要考虑一个问题:这条记录真的会被更新覆盖吗?显然是不会的:

  • 执行器调用引擎接口查询name='小明’的数据,如果有缓存会用到缓存
  • 执行器拿到查询的数据,age字段改为18得到新的数据行,然后调用引擎API接口,写入新数据并将原数据标记删除,InnoDB引擎把数据保存在内存中,同时记录redo log,此时redo log进入prepare状态,然后告诉执行器,执行完成了,随时可以提交
  • 执行器收到通知后记录binlog,然后调用引擎接口,提交redo log 为提交状态
  • 更新完成
    上面用到了MySQL的binlog和InnoDB的redo log,看似很繁琐,但是从反方向来看一下就明白其用意了:
  • 先写redo log直接提交,假设此时机器挂了,然后写binlog失败,那么机器重启后,这台机器会通过redo log恢复数据,但是这个时候bingog并没有记录该数据,后续进行机器备份的时候,就会丢失这一条数据,同时主从同步也会丢失这一条数据
  • 先写binlog,假设机器异常重启了,由于没有redo log,本机是无法恢复这一条记录的,但是binlog又有记录,那么和上面同样的道理,就会产生数据不一致的情况

如果采用redo log 两阶段提交的方式就不一样了,写完binglog后,然后再提交redo log就会防止出现上述的问题,从而保证了数据的一致性。那么问题来了,有没有一个极端的情况呢?假设redo log 处于预提交状态,binglog也已经写完了,这个时候发生了异常重启会怎么样呢? 这个就要依赖于mysql的处理机制了:

  • 判断redo log 是否完整,如果判断是完整的,就立即提交。
  • 如果redo log 只是预提交但不是commit状态,这个时候就会去判断binlog是否完整,如果完整就提交 redo log, 不完整就回滚事务。
2-3、删除:

执行delete语句时,Innodb作删除标记;如果后续插入数据时,定位到这个数据页上有可能继续复用这个位置
同理,如果一个数据页上的数据都被删除了, 那么这个数据页也是可以被复用的

数据空洞

Inndb的标记删除,是不会主动回收表空间的,这些没有被复用的空间就造成了“空洞”
插入数据的时候,如果是随机插入,就有可能形成数据页分裂,就会造成数据页的末尾形成空洞
更新值可以理解为删除一个旧值, 插入一个新值。 同样会造成空洞
一个数据表经过大量的增删改后,都是有可能造成空洞的, 通过重建表可进行回收

其实到这里基本思路已经可以清晰的看到了,接下来就是实际操作来证实一些猜想,可能有遗漏的欢迎评论区补全 三、优化

根据上面的执行流程可以得出:更新操作离不开查询,所以就从查询入手进行优化

3-1、数据库表结构的优化

1.选取最适用的字段属性

  • 使用可以存下你的数据的最小的数据类型
  • 使用简单的数据类型。Int要比varchar类型在mysql处理上简单
  • 尽可能的使用not null定义字段
  • 尽量少用text类型

2.表的范式化和反范式化

  • 不符合第三范式要求的表存在数据冗余,可能会导致更新插入删除数据异常情况,反范式化是指为了查询效率和考虑把原本符合第三范式的表适当的增加冗余,以达到优化查询效率的目的,反范式化是一种以空间来换取时间的操作。
    例:
表名
用户表id名称电话
签到表1id用户id签到时间
签到表2id用户id签到时间用户名称用户电话

签到表2就是签到表1的反范式化,SELECT a.*,b.* FROM 签到表1 a,用户表 b WHERe b.id=a.用户id→SELECt * FROM 签到表2同样的需要查询一条完整的签到信息时,查询反范式化的签到表可以不用查询两张表,以提高效率

3.表的垂直拆分

  • 基础数据不常更新的放在一张表中。
  • 把大字段独立存放到一个表中。
  • 不常用的字段单独存放到一个表中。

4.水平拆分

  • 表的水平拆分是为了解决单表的数据量过大问题,水平拆分的表每个表的结构都是完全一致的,可以根据时间进行分片分表
3-2、SQL语句优化
  • 尽量避免使用子查询
    SELECt * FROM t1 WHERe ID IN (SELECt ID FROM t2 WHERe NAME='小明')在MySQL5.5版本,执行器会先查外表再匹配内表,而不是先查t2,当t1数据量大时,速度会非常慢。
    在MariaDB10/MySQL5.6版本里,采用join关联方式进行了优化:SELECt t1.* FROM t1 JOIN t2 ON t1.ID = t2.ID WHERe t2.NAME='小明',但是优化只对SELECt语句有效,UPDATE或DELETE子查询无效,所以还是要避免使用子查询
  • 避免对索引列使用函数操作
    对索引列使用函数操作会导致全表扫描
  • 单字段多值匹配用IN不用OR,连续数值用BETWEEN不用IN
  • LIKE避免单独使用左通配符
  • 读取适当的记录LIMIT M,N,而不要读多余的记录
    在ID是连续自增的前提下,SELECT ID,NAME,AGE FROM tb_user WHERe ID >120000 LIMIT 20 执行优于 SELECt ID,NAME,AGE FROM tb_user LIMIT 120001,20
  • 避免数据类型转换
    NAME字段为varchar(20)的前提下,SELECt ID,NAME,AGE FROM tb_user WHERe NAME=123需要先将数字类型的123转换为字符类型’123’再去匹配
  • 避免随机取记录,使用RAND()函数会导致全表扫描
  • 使用NOT等负向查询条件会导致全表查
  • 尽量不用select * from

节省数据库解析*代表哪些字段的过程,降低开销;可能存在不必要的列,传输过程有不必要的性能损耗;

  • 区分in和exists
    SELECt * FROM tb_1 WHERe ID IN (SELECt ID FROM tb_2)与SELECt * FROM tb_1 WHERe EXISTS( (SELECt * FROM tb_2 WHERe tb1.ID=tb2.ID)执行结果是一样的,区分in和exists主要是造成了驱动顺序的改变(这是性能变化的关键),如果是exists,那么以外层表为驱动表,先被访问,如果是IN,那么先执行子查询。所以IN适合于外表大而内表小的情况;EXISTS适合于外表小而内表大的情况
  • 优化GROUP BY语句
    排序时尽量使用索引字段,可以通过EXPLAIN查看执行计划里有没有Using temporary 和 Using filesort
3-3、索引的优化
  • EXPLAIN优化查询检测
    EXPLAIN可以分析SQL,判断SQL如何使用索引处理SELECt语句以及表的连接,可以帮助调整SQL语句以写出更优化的查询。
字段名含义值或备注
select_type查询类型

1.SIMPLE(简单SELECT,不使用UNIOn或子查询等) ;

2.PRIMARY(子查询中最外层查询,查询中若包含任何复杂的子部分,最外层的select被标记为PRIMARY)

3.UNIOn(UNIOn中的第二个或后面的SELECT语句)

4.DEPENDENT UNIOn(UNIOn中的第二个或后面的SELECT语句,取决于外面的查询)

5.UNIOn RESULT(UNIOn的结果,union语句中第二个select开始后面所有select)

6.SUBQUERY(子查询中的第一个SELECT,结果不依赖于外部查询)

7.DEPENDENT SUBQUERY(子查询中的第一个SELECT,依赖于外部查询)

8.DERIVED(派生表的SELECT, FROM子句的子查询)

9.UNCACHEABLE SUBQUERY(一个子查询的结果不能被缓存,必须重新评估外链接的第一行)

table输出结果集的表显示这一步所访问数据库中表名称(显示这一行的数据是关于哪张表的),有时不是真实的表名字,可能是简称,也可能是第几步执行的结果的简称
tpartitions匹配的分区
type表示表的连接类型

1.ALL:Full Table Scan, MySQL将遍历全表以找到匹配的行

2.index: Full Index Scan,index与ALL区别为index类型只遍历索引树

3.range:只检索给定范围的行,使用一个索引来选择行

4.ref: 表示上述表的连接匹配条件,即哪些列或常量被用于查找索引列上的值

5.eq_ref: 类似ref,区别就在使用的索引是唯一索引,对于每个索引键值,表中只有一条记录匹配,简单来说,就是多表连接中使用primary key或者 unique key作为关联条件

6.const、system: 当MySQL对查询某部分进行优化,并转换为一个常量时,使用这些类型访问。如将主键置于where列表中,MySQL就能将该查询转换为一个常量,system是const类型的特例,当查询的表只有一行的情况下,使用system

7.NULL: MySQL在优化过程中分解语句,执行时甚至不用访问表或索引,例如从一个索引列里选取最小值可以通过单独索引查找完成。

possible_keys查询时,可能使用的索引
key实际使用的索引
key_len索引字段的长度(不损失精确性的情况下,长度越短越好)
ref列与索引的比较(哪些列或常量被用于查找索引列上的值)
rows扫描出的行数(估算的行数)
filtered按表条件过滤的行百分比
Extra执行情况的描述和说明

1.Using where:不用读取表中所有信息,仅通过索引就可以获取所需数据,这发生在对表的全部的请求列都是同一个索引的部分的时候,表示mysql服务器将在存储引擎检索行后再进行过滤

2.Using temporary:表示MySQL需要使用临时表来存储结果集,常见于排序和分组查询,常见 group by ; order by

3.Using filesort:当Query中包含 order by 操作,而且无法利用索引完成的排序操作称为“文件排序”

  • 索引的类型
    UNIQUE唯一索引
    不可以出现相同的值,可以有NULL值。
    INDEX普通索引
    允许出现相同的索引内容。
    PRIMARY KEY主键索引
    不允许出现相同的值,且不能为NULL值,一个表只能有一个primary_key索引。
    fulltext index 全文索引
    上述三种索引都是针对列的值发挥作用,但全文索引,可以针对值中的某个单词,比如一篇文章中的某个词,然而并没有什么用,因为只有myisam以及英文支持,并且效率极低,但是可以用coreseek和xunsearch等第三方应用来完成这个需求。

  • 创建索引的技巧
    1.维度高的列创建索引,数据列中不重复值出现的个数,这个数量越高,维度就越高;
    比如性别和年龄,那年龄的维度就高于性别;且性别这样的列不适合创建索引,因为维度过低。
    2.对 where,on,group by,order by 中出现的列创建索引。
    3.对较小的数据列使用索引,这样会使索引文件更小,同时内存中也可以装载更多的索引键。
    4.为较长的字符串使用前缀索引。
    5.不要过多创建索引,除了增加额外的磁盘空间外,对于DML操作的速度影响很大,因为其每增删改一次就得从新建立索引。
    6.使用组合索引,可以减少文件索引大小,在使用时速度要优于多个单列索引。

3-4、系统配置的优化
[client]
port = 3306
socket = /tmp/mysql.sock
[mysqld]
port = 3306
socket = /tmp/mysql.sock
basedir = /usr/local/mysql
datadir = /data/mysql
pid-file = /data/mysql/mysql.pid
user = mysql
bind-address = 0.0.0.0
server-id = 1 #表示是本机的序号为1,一般来讲就是master的意思
skip-name-resolve
# 禁止MySQL对外部连接进行DNS解析,使用这一选项可以消除MySQL进行DNS解析的时间。但需要注意,如果开启该选项,
# 则所有远程主机连接授权都要使用IP地址方式,否则MySQL将无法正常处理连接请求
#skip-networking
back_log = 600
# MySQL能有的连接数量。当主要MySQL线程在一个很短时间内得到非常多的连接请求,这就起作用,
# 然后主线程花些时间(尽管很短)检查连接并且启动一个新线程。back_log值指出在MySQL暂时停止回答新请求之前的短时间内多少个请求可以被存在堆栈中。
# 如果期望在一个短时间内有很多连接,你需要增加它。也就是说,如果MySQL的连接数据达到max_connections时,新来的请求将会被存在堆栈中,
# 以等待某一连接释放资源,该堆栈的数量即back_log,如果等待连接的数量超过back_log,将不被授予连接资源。
# 另外,这值(back_log)限于您的操作系统对到来的TCP/IP连接的侦听队列的大小。
# 你的操作系统在这个队列大小上有它自己的限制(可以检查你的OS文档找出这个变量的最大值),试图设定back_log高于你的操作系统的限制将是无效的。
max_connections = 1000
# MySQL的最大连接数,如果服务器的并发连接请求量比较大,建议调高此值,以增加并行连接数量,当然这建立在机器能支撑的情况下,因为如果连接数越多,介于MySQL会为每个连接提供连接缓冲区,就会开销越多的内存,所以要适当调整该值,不能盲目提高设值。可以过'conn%'通配符查看当前状态的连接数量,以定夺该值的大小。
max_connect_errors = 6000
# 对于同一主机,如果有超出该参数值个数的中断错误连接,则该主机将被禁止连接。如需对该主机进行解禁,执行:FLUSH HOST。
open_files_limit = 65535
# MySQL打开的文件描述符限制,默认最小1024;当open_files_limit没有被配置的时候,比较max_connections*5和ulimit -n的值,哪个大用哪个,
# 当open_file_limit被配置的时候,比较open_files_limit和max_connections*5的值,哪个大用哪个。
table_open_cache = 128
# MySQL每打开一个表,都会读入一些数据到table_open_cache缓存中,当MySQL在这个缓存中找不到相应信息时,才会去磁盘上读取。默认值64
# 假定系统有200个并发连接,则需将此参数设置为200*N(N为每个连接所需的文件描述符数目);
# 当把table_open_cache设置为很大时,如果系统处理不了那么多文件描述符,那么就会出现客户端失效,连接不上
max_allowed_packet = 4M
# 接受的数据包大小;增加该变量的值十分安全,这是因为仅当需要时才会分配额外内存。例如,仅当你发出长查询或MySQLd必须返回大的结果行时MySQLd才会分配更多内存。
# 该变量之所以取较小默认值是一种预防措施,以捕获客户端和服务器之间的错误信息包,并确保不会因偶然使用大的信息包而导致内存溢出。
binlog_cache_size = 1M
# 一个事务,在没有提交的时候,产生的日志,记录到Cache中;等到事务提交需要提交的时候,则把日志持久化到磁盘。默认binlog_cache_size大小32K
max_heap_table_size = 8M
# 定义了用户可以创建的内存表(memory table)的大小。这个值用来计算内存表的最大行数值。这个变量支持动态改变
tmp_table_size = 16M
# MySQL的heap(堆积)表缓冲大小。所有联合在一个DML指令内完成,并且大多数联合甚至可以不用临时表即可以完成。
# 大多数临时表是基于内存的(HEAP)表。具有大的记录长度的临时表 (所有列的长度的和)或包含BLOB列的表存储在硬盘上。
# 如果某个内部heap(堆积)表大小超过tmp_table_size,MySQL可以根据需要自动将内存中的heap表改为基于硬盘的MyISAM表。还可以通过设置tmp_table_size选项来增加临时表的大小。也就是说,如果调高该值,MySQL同时将增加heap表的大小,可达到提高联接查询速度的效果
read_buffer_size = 2M
# MySQL读入缓冲区大小。对表进行顺序扫描的请求将分配一个读入缓冲区,MySQL会为它分配一段内存缓冲区。read_buffer_size变量控制这一缓冲区的大小。
# 如果对表的顺序扫描请求非常频繁,并且你认为频繁扫描进行得太慢,可以通过增加该变量值以及内存缓冲区大小提高其性能
read_rnd_buffer_size = 8M
# MySQL的随机读缓冲区大小。当按任意顺序读取行时(例如,按照排序顺序),将分配一个随机读缓存区。进行排序查询时,
# MySQL会首先扫描一遍该缓冲,以避免磁盘搜索,提高查询速度,如果需要排序大量数据,可适当调高该值。但MySQL会为每个客户连接发放该缓冲空间,所以应尽量适当设置该值,以避免内存开销过大
sort_buffer_size = 8M
# MySQL执行排序使用的缓冲大小。如果想要增加ORDER BY的速度,首先看是否可以让MySQL使用索引而不是额外的排序阶段。
# 如果不能,可以尝试增加sort_buffer_size变量的大小
join_buffer_size = 8M
# 联合查询操作所能使用的缓冲区大小,和sort_buffer_size一样,该参数对应的分配内存也是每连接独享
thread_cache_size = 8
# 这个值(默认8)表示可以重新利用保存在缓存中线程的数量,当断开连接时如果缓存中还有空间,那么客户端的线程将被放到缓存中,
# 如果线程重新被请求,那么请求将从缓存中读取,如果缓存中是空的或者是新的请求,那么这个线程将被重新创建,如果有很多新的线程,
# 增加这个值可以改善系统性能.通过比较Connections和Threads_created状态的变量,可以看到这个变量的作用。(–>表示要调整的值)
# 根据物理内存设置规则如下:
# 1G  —> 8
# 2G  —> 16
# 3G  —> 32
# 大于3G  —> 64
query_cache_size = 8M
#MySQL的查询缓冲大小(从4.0.1开始,MySQL提供了查询缓冲机制)使用查询缓冲,MySQL将SELECT语句和查询结果存放在缓冲区中,
# 今后对于同样的SELECT语句(区分大小写),将直接从缓冲区中读取结果。根据MySQL用户手册,使用查询缓冲最多可以达到238%的效率。
# 通过检查状态值'Qcache_%',可以知道query_cache_size设置是否合理:如果Qcache_lowmem_prunes的值非常大,则表明经常出现缓冲不够的情况,
# 如果Qcache_hits的值也非常大,则表明查询缓冲使用非常频繁,此时需要增加缓冲大小;如果Qcache_hits的值不大,则表明你的查询重复率很低,
# 这种情况下使用查询缓冲反而会影响效率,那么可以考虑不用查询缓冲。此外,在SELECT语句中加入SQL_NO_CACHE可以明确表示不使用查询缓冲
query_cache_limit = 2M
#指定单个查询能够使用的缓冲区大小,默认1M
key_buffer_size = 4M
#指定用于索引的缓冲区大小,增加它可得到更好处理的索引(对所有读和多重写),到你能负担得起那样多。如果你使它太大,
# 系统将开始换页并且真的变慢了。对于内存在4GB左右的服务器该参数可设置为384M或512M。通过检查状态值Key_read_requests和Key_reads,
# 可以知道key_buffer_size设置是否合理。比例key_reads/key_read_requests应该尽可能的低,
# 至少是1:100,1:1000更好(上述状态值可以使用SHOW STATUS LIKE 'key_read%'获得)。注意:该参数值设置的过大反而会是服务器整体效率降低
ft_min_word_len = 4
# 分词词汇最小长度,默认4
transaction_isolation = REPEATABLE-READ
# MySQL支持4种事务隔离级别,他们分别是:
# READ-UNCOMMITTED, READ-COMMITTED, REPEATABLE-READ, SERIALIZABLE.
# 如没有指定,MySQL默认采用的是REPEATABLE-READ,ORACLE默认的是READ-COMMITTED
log_bin = mysql-bin
binlog_format = mixed
expire_logs_days = 30 #超过30天的binlog删除
log_error = /data/mysql/mysql-error.log #错误日志路径
slow_query_log = 1
long_query_time = 1 #慢查询时间 超过1秒则为慢查询
slow_query_log_file = /data/mysql/mysql-slow.log
performance_schema = 0
explicit_defaults_for_timestamp
#lower_case_table_names = 1 #不区分大小写
skip-external-locking #MySQL选项以避免外部锁定。该选项默认开启
default-storage-engine = InnoDB #默认存储引擎
innodb_file_per_table = 1
# InnoDB为独立表空间模式,每个数据库的每个表都会生成一个数据空间
# 独立表空间优点:
# 1.每个表都有自已独立的表空间。
# 2.每个表的数据和索引都会存在自已的表空间中。
# 3.可以实现单表在不同的数据库中移动。
# 4.空间可以回收(除drop table操作处,表空不能自已回收)
# 缺点:
# 单表增加过大,如超过100G
# 结论:
# 共享表空间在Insert操作上少有优势。其它都没独立表空间表现好。当启用独立表空间时,请合理调整:innodb_open_files
innodb_open_files = 500
# 限制Innodb能打开的表的数据,如果库里的表特别多的情况,请增加这个。这个值默认是300
innodb_buffer_pool_size = 64M
# InnoDB使用一个缓冲池来保存索引和原始数据, 不像MyISAM.
# 这里你设置越大,你在存取表里面数据时所需要的磁盘I/O越少.
# 在一个独立使用的数据库服务器上,你可以设置这个变量到服务器物理内存大小的80%
# 不要设置过大,否则,由于物理内存的竞争可能导致操作系统的换页颠簸.
# 注意在32位系统上你每个进程可能被限制在 2-3.5G 用户层面内存限制,
# 所以不要设置的太高.
innodb_write_io_threads = 4
innodb_read_io_threads = 4
# innodb使用后台线程处理数据页上的读写 I/O(输入输出)请求,根据你的 CPU 核数来更改,默认是4
# 注:这两个参数不支持动态改变,需要把该参数加入到my.cnf里,修改完后重启MySQL服务,允许值的范围从 1-64
innodb_thread_concurrency = 0
# 默认设置为 0,表示不限制并发数,这里推荐设置为0,更好去发挥CPU多核处理能力,提高并发量
innodb_purge_threads = 1
# InnoDB中的清除操作是一类定期回收无用数据的操作。在之前的几个版本中,清除操作是主线程的一部分,这意味着运行时它可能会堵塞其它的数据库操作。
# 从MySQL5.5.X版本开始,该操作运行于独立的线程中,并支持更多的并发数。用户可通过设置innodb_purge_threads配置参数来选择清除操作是否使用单
# 独线程,默认情况下参数设置为0(不使用单独线程),设置为 1 时表示使用单独的清除线程。建议为1
innodb_flush_log_at_trx_commit = 2
# 0:如果innodb_flush_log_at_trx_commit的值为0,log buffer每秒就会被刷写日志文件到磁盘,提交事务的时候不做任何操作(执行是由mysql的master thread线程来执行的。
# 主线程中每秒会将重做日志缓冲写入磁盘的重做日志文件(REDO LOG)中。不论事务是否已经提交)默认的日志文件是ib_logfile0,ib_logfile1
# 1:当设为默认值1的时候,每次提交事务的时候,都会将log buffer刷写到日志。
# 2:如果设为2,每次提交事务都会写日志,但并不会执行刷的操作。每秒定时会刷到日志文件。要注意的是,并不能保证100%每秒一定都会刷到磁盘,这要取决于进程的调度。
# 每次事务提交的时候将数据写入事务日志,而这里的写入仅是调用了文件系统的写入操作,而文件系统是有 缓存的,所以这个写入并不能保证数据已经写入到物理磁盘
# 默认值1是为了保证完整的ACID。当然,你可以将这个配置项设为1以外的值来换取更高的性能,但是在系统崩溃的时候,你将会丢失1秒的数据。
# 设为0的话,mysqld进程崩溃的时候,就会丢失最后1秒的事务。设为2,只有在操作系统崩溃或者断电的时候才会丢失最后1秒的数据。InnoDB在做恢复的时候会忽略这个值。
# 总结
# 设为1当然是最安全的,但性能页是最差的(相对其他两个参数而言,但不是不能接受)。如果对数据一致性和完整性要求不高,完全可以设为2,如果只最求性能,例如高并发写的日志服务器,设为0来获得更高性能
innodb_log_buffer_size = 2M
# 此参数确定些日志文件所用的内存大小,以M为单位。缓冲区更大能提高性能,但意外的故障将会丢失数据。MySQL开发人员建议设置为1-8M之间
innodb_log_file_size = 32M
# 此参数确定数据日志文件的大小,更大的设置可以提高性能,但也会增加恢复故障数据库所需的时间
innodb_log_files_in_group = 3
# 为提高性能,MySQL可以以循环方式将日志文件写到多个文件。推荐设置为3
innodb_max_dirty_pages_pct = 90
# innodb主线程刷新缓存池中的数据,使脏数据比例小于90%
innodb_lock_wait_timeout = 120 
# InnoDB事务在被回滚之前可以等待一个锁定的超时秒数。InnoDB在它自己的锁定表中自动检测事务死锁并且回滚事务。InnoDB用LOCK TABLES语句注意到锁定设置。默认值是50秒
bulk_insert_buffer_size = 8M
# 批量插入缓存大小, 这个参数是针对MyISAM存储引擎来说的。适用于在一次性插入100-1000+条记录时, 提高效率。默认值是8M。可以针对数据量的大小,翻倍增加。
myisam_sort_buffer_size = 8M
# MyISAM设置恢复表之时使用的缓冲区的尺寸,当在REPAIR TABLE或用CREATE INDEX创建索引或ALTER TABLE过程中排序 MyISAM索引分配的缓冲区
myisam_max_sort_file_size = 10G
# 如果临时文件会变得超过索引,不要使用快速排序索引方法来创建一个索引。注释:这个参数以字节的形式给出
myisam_repair_threads = 1
# 如果该值大于1,在Repair by sorting过程中并行创建MyISAM表索引(每个索引在自己的线程内)  
interactive_timeout = 28800
# 服务器关闭交互式连接前等待活动的秒数。交互式客户端定义为在mysql_real_connect()中使用CLIENT_INTERACTIVE选项的客户端。默认值:28800秒(8小时)
wait_timeout = 28800
# 服务器关闭非交互连接之前等待活动的秒数。在线程启动时,根据全局wait_timeout值或全局interactive_timeout值初始化会话wait_timeout值,
# 取决于客户端类型(由mysql_real_connect()的连接选项CLIENT_INTERACTIVE定义)。参数默认值:28800秒(8小时)
# MySQL服务器所支持的最大连接数是有上限的,因为每个连接的建立都会消耗内存,因此我们希望客户端在连接到MySQL Server处理完相应的操作后,
# 应该断开连接并释放占用的内存。如果你的MySQL Server有大量的闲置连接,他们不仅会白白消耗内存,而且如果连接一直在累加而不断开,
# 最终肯定会达到MySQL Server的连接上限数,这会报'too many connections'的错误。对于wait_timeout的值设定,应该根据系统的运行情况来判断。
# 在系统运行一段时间后,可以通过show processlist命令查看当前系统的连接状态,如果发现有大量的sleep状态的连接进程,则说明该参数设置的过大,
# 可以进行适当的调整小些。要同时设置interactive_timeout和wait_timeout才会生效。
[mysqldump]
quick
max_allowed_packet = 16M #服务器发送和接受的最大包长度
[myisamchk]
key_buffer_size = 8M
sort_buffer_size = 8M
read_buffer = 4M
write_buffer = 4M

3-5、硬件的优化

升级服务器硬件,扩展服务器集群等等

转载请注明:文章转载自 www.051e.com
本文地址:http://www.051e.com/it/1057979.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 ©2023-2025 051e.com

ICP备案号:京ICP备12030808号