Mysql的Locked状态

2024-02-24 19:58
文章标签 mysql 状态 database locked

本文主要是介绍Mysql的Locked状态,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


  

最近搬迁服务器也遇到一个问题,一个ecshop的站点,访问速度很慢,以前也没有这种情况发生,现在出现这么慢的情况确实让人疑惑,换了另外一个服务器也还是这样,可是访问的人也不多呢,带宽也不差,所以纳闷…

    多搜索一下资料看看,似乎找到了出路,可以不用再尝试换服务器了。

    我把表类型换为innodb类型后,情况就好很多了,请参考后面的资料。

    sql语句类似 ALTER p aslibra TYPE = innodb ; 很简单

    其实碰到这个问题我就看了一下数据库的进程,发现有很多locked状态的查询语句,比较奇怪为何会这样,为什么会出现这样的情况的,这个情况很可能是读写限制所导致的,也可能是读写速度有关,可是不理解为何一直锁定而不放开,导致页面等待时间很长。况且读写也要求不高,带宽访问人数等都并不高呢?

    换了一个数据库还是不行,看来得郁闷一阵了,搬服务器真能够让人变得暴躁,起码我就不耐烦了,搬来搬去的折磨的要死,就好像给你两个瓶子,从A瓶子倒东西到B瓶子,然后复制一份到A,把A搬走,A稳定后,又开始把A的东西挪到B,反复折磨人的意志。

    查了些资料,记录一下,希望以后借鉴:

    引用最近有个项目网站访问很慢。上去查了查,发现cpu经常100%,运行vmstat看了看,发现瓶颈不在cpu,不在磁盘上。系统是java+mysql,发现这两个东西也几乎不占用cpu,都是排在很后面的。

    难道是网络问题,查了一下也不是,那么就是mysql内部问题。运行 SHOW PROCESSLIST,发现很多Locked,难道就是这个问题。因为一开始就把mysql设置成MyIasm的格式,因为这个速度比较快,不过这种格式是表锁定的,看来就是这里。于是改了程序中某些频繁更新的操作,做成cache,然后更新上去,现在速度好了。不过下来还是需要把数据库换成innodb格式。

    由此得出结论,MyIasm适合读远大于写的情况,比如CMS之类的,而读写频繁的可能要innodb方式的。javaeye的robbin把数据库从myiasm换成innodb据说性能好了太多,看来也是这个问题。

    2 《轻松掌握MySQL数据库锁机制的相关原理》

    MySQL 5.1支持对MyISAM和MEMORY表进行表级锁定,对BDB表进行页级锁定,对InnoDB 表进行行级锁定。在许多情况下,可以根据培训猜测应用程序使用哪类锁定类型最好,但一般很难说出某个给出的锁类型就比另一个好。一切取决于应用程序,应用程序的不同部分可能需要不同的锁类型。为了确定是否想要使用行级锁定的存储引擎,应看看应用程序做什么并且混合使用什么样的选择和更新语句。例如,大多数Web应用程序执行许多选择,而很少进行删除,只对关键字的值进行更新,并且只插入少量具体的表。基本MySQL MyISAM设置已经调节得很好。

    在MySQL中对于使用表级锁定的存储引擎,表锁定时不会死锁的。这通常总是在一个查询开始时立即请求所有必要的锁定并且总是以同样的顺序锁定表来管理。(myisam 以有序的顺序去所表,inodb 没有顺序的去所表,所以会产生死锁)

    对WRITE,MySQL使用的表锁定方法原理如下:

    ◆ 如果在表上没有锁,在它上面放一个写锁。

    ◆否则,把锁定请求放在写锁定队列中。

    对READ,MySQL使用的锁定方法原理如下:

    ◆如果在表上没有写锁定,把一个读锁定放在它上面。

    ◆否则,把锁请求放在读锁定队列中。

    当一个锁定被释放时,锁定可被写锁定队列中的线程得到,然后是读锁定队列中的线程。

    这意味着,如果你在一个表上有许多更新,SELECT语句将等待直到没有更多的更新。

    如果INSERT 语句不冲突,可以自由为MyISAM 表混合并行的INSERT 和SELECT 语句而不需要锁定。

    InnoDB 使用行锁定,BDB 使用页锁定。对于这两种存储引擎,都可能存在死锁。这是因为,在SQL语句处理期间,InnoDB 自动获得行锁定,BDB 获得页锁定,而不是在事务启动时获得。

也就是说对MyISAM表的读操作,不会阻塞其他用户对同一表的读请求,但会阻塞对同一表的写请求;对 MyISAM表的写操作,则会阻塞其他用户对同一表的读和写操作;MyISAM表的读操作与写操作之间,以及写操作之间是串行的!

优先级:update --->  (删除与select、insert 优先级相同)

当表为 myisam 时 频繁插入的数据会出现 table lock   这就是锁表


行级锁定的优点:

    · 当在许多线程中访问不同的行时只存在少量锁定冲突。

    · 回滚时只有少量的更改。

    · 可以长时间锁定单一的行。

    行级锁定的缺点:

    · 比页级或表级锁定占用更多的内存。

    · 当在表的大部分中使用时,比页级或表级锁定速度慢,因为你必须获取更多的锁。

    · 如果你在大部分数据上经常进行 GROUP BY 操作或者必须经常扫描整个表,比其它锁定明显慢很多。

    · 用高级别锁定,通过支持不同的类型锁定,你也可以很容易地调节应用程序,因为其锁成本小于行级锁定。

    在以下情况下,表锁定优先于页级或行级锁定:

    · 表的大部分语句用于读取。

    · 对严格的关键字进行读取和更新,你可以更新或删除可以用单一的读取的关键字来提取的一行:

    ? UPDATE tbl_name SET column = value WHERE unique_key_col = key_value ;

    ? DELETE FROM tbl_name WHERE unique_key_col = key_value ;

    · SELECT 结合并行的INSERT 语句,并且只有很少的UPDATE或 DELETE 语句。

    · 在整个表上有许多扫描或 GROUP BY 操作,没有任何写操作。

    不同于行级或页级锁定的选项:

    · 版本(例如,为并行的插入在MySQL中使用的技术),其中可以一个写操作,同时有许多读取操作。这明数据库或表支持数据依赖的不同视图,取决于访问何时开始。其它共同的术语是"时间跟踪"、"写复制"或者"按需复制".

    · 按需复制在许多情况下优先于页级或行级锁定。然而,在最坏的情况下,它可能比使用常规锁定使用多的内存。

    · 除了行级锁定外,你可以使用应用程序级锁定,例如在MySQL中使用GET_LOCK()和RELEASE_LOCK()。这些是建议性锁定,它们只能在运行良好的应用程序中工作。

    为达到最高锁定速度,除InnoDB 和BDB 之外,对所有存储引擎,MySQL使用表锁定(而不是页、行或者列锁定)。对于InnoDB 和BDB 表,如果你用LOCK TABLES显式锁定表,MySQL只使用表锁定;如果你不使用LOCK TABLES,因为 InnoDB 使用自动行级锁定而BDB 使用页级锁定来保证事务隔离。

    但是对于大表,对于大多数应用程序,表锁定比行锁定更好,但存在部分缺陷。表锁定使许多线程同时从一个表中进行读取操作,但如果一个线程想要对表进行写操作,它必须首先获得独占访问。更新期间,所有其它想要访问该表的线程必须等待直到更新完成。

    表更新通常情况认为比表检索更重要,因此给予它们更高的优先级。这应确保更新一个表的活动不能"饿死",即使该表上有很繁重的SELECT 活动。

    表锁定在这种情况下会造成问题,例如当线程正等待,因为硬盘已满并且在线程可以处理之前必须有空闲空间。在这种情况下,所有想要访问出现问题的表的线程也被设置成等待状态,直到有更多的硬盘空间可用。

    表锁定在下面的情况下也存在问题:

    · 一个客户发出长时间运行的查询。

    · 然后,另一个客户对同一个表进行更新。该客户必须等待直到SELECT完成。

    · 另一个客户对同一个表上发出了另一个 SELECT 语句。因为UPDATE比 SELECT 优先级高,该SELECT 语句等待UPDATE完成,并且等待第1个 SELECT 完成。

    下面描述了一些方法来避免或减少表锁定造成的竞争:

    · 试图使 SELECT 语句运行得更快。可能必须创建一些摘要(summary)表做到这点。

    · 用--low-priority-updates启动mysqld.这将给所有更新(修改)一个表的语句以比SELECT语句低的优先级。在这种情况下,在先前情形的第2个SELECT语句将在UPDATE语句前执行,而不需要等候第1个 SELECT 完成。

    · 可以使用SET_UPDATES=1语句指定具体连接中的所有更新应使用低优先级。

    · 可以用LOW_PRIORITY属性给与一个特定的INSERT、UPDATE或DELETE语句较低优先级。

    · 可以用HIGH_PRIORITY属性给与一个特定的SELECT语句较高优先级。

    · 为max_write_lock_count系统变量指定一个低值来启动mysqld来强制MySQL在具体数量的插入完成后临时提高所有等待一个表的SELECT 语句的优先级。这样允许在一定数量的WRITE锁定后给出READ锁定。

    · 如果你有关于INSERT结合SELECT的问题,切换到使用新的MyISAM表,因为它们支持并发的SELECT和INSERT.

    · 如果你对同一个表混合插入和删除,INSERT DELAYED将会有很大的帮助。

    · 如果你对同一个表混合使用 SELECT 和DELETE 语句出现问题,DELETE 的LIMIT 选项可以有所帮助。

    · 对 SELECT 语句使用SQL_BUFFER_RESULT可以帮助使表锁定时间变短。

    · 可以更改mysys/thr_lock.c中的锁代码以使用单一的队列。在这种情况下,写锁定和读锁定将具有相同的优先级,对一些应用程序会有帮助。

    这里是一些MySQL中表锁定相关的技巧:

    · 如果不混合更新与需要在同一个表中检查许多行的选择,可以进行并行操作。

    · 可以使用 LOCK TABLES 来提高速度,因为在一个锁定中进行许多更新比没有锁定的更新要快得多。将表中的内容切分为几个表也可以有所帮助。

    · 如果在MySQL中表锁定时遇到速度问题,可以将表转换为 InnoDB 或BDB 表来提高性能


解决方案大概有如下几种:
  
  1. MyISAM存储引擎有一个系统变量concurrent_insert,专门用以控制其并发插入的行为,其值分别可以为0、1或2。
    • 0 不允许并发操作
    • 1 如果MyISAM表中没有空洞(即表的中间没有被删除的行),MyISAM允许在一个进程读表的同时,另一个进程从表尾插入记录。这也是MySQL的默认设置。
    • 2 无论MyISAM表中有没有空洞,都允许在表尾并发插入记录
  2. 使用--low-priority-updates启用mysqld。这将给所有更新(修改)一个表的语句以比SELECT语句低的优先级。在这种情况下,在先前情形的最后的SELECT语句将在INSERT语句前执行。
  3. 为max_write_lock_count设置一个低值,使得在一定数量的WRITE锁定后,给出READ锁定
  4. 使用LOW_PRIORITY属性给于一个特定的INSERT,UPDATE或DELETE较低的优先级
  5. 使用HIGH_PRIORITY属性给于一个特定的SELECT
  6. 使用INSERT DELAYED语句
可以通过show status like 'table%'检查Table_locks_immediate和Table_locks_waited发现Table_locks_waited看是否偏 大。如果出问题的表是MyISAM,分析大概是MyISAM的表锁导致。


转自:http://hi.baidu.com/lqk12/item/0271953f53135323b3c0c5ee

这篇关于Mysql的Locked状态的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/743202

相关文章

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

oracle DBMS_SQL.PARSE的使用方法和示例

《oracleDBMS_SQL.PARSE的使用方法和示例》DBMS_SQL是Oracle数据库中的一个强大包,用于动态构建和执行SQL语句,DBMS_SQL.PARSE过程解析SQL语句或PL/S... 目录语法示例注意事项DBMS_SQL 是 oracle 数据库中的一个强大包,它允许动态地构建和执行

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

在MySQL执行UPDATE语句时遇到的错误1175的解决方案

《在MySQL执行UPDATE语句时遇到的错误1175的解决方案》MySQL安全更新模式(SafeUpdateMode)限制了UPDATE和DELETE操作,要求使用WHERE子句时必须基于主键或索引... mysql 中遇到的 Error Code: 1175 是由于启用了 安全更新模式(Safe Upd

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySql死锁怎么排查的方法实现

《MySql死锁怎么排查的方法实现》本文主要介绍了MySql死锁怎么排查的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录前言一、死锁排查方法1. 查看死锁日志方法 1:启用死锁日志输出方法 2:检查 mysql 错误

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

MySQL修改密码的四种实现方式

《MySQL修改密码的四种实现方式》文章主要介绍了如何使用命令行工具修改MySQL密码,包括使用`setpassword`命令和`mysqladmin`命令,此外,还详细描述了忘记密码时的处理方法,包... 目录mysql修改密码四种方式一、set password命令二、使用mysqladmin三、修改u