php mysql事务处理

事务

BEGIN;
COMMIT;
一段代码, 修改当前表的两个字段

先查找出来中,保存到变量里面,然后相加放到第二个字段的value 里面

 

 代码如下 复制代码
BEGIN;
SELECT value into  @short_destiption  FROM `catalog_product_entity_text` WHERE entity_id = 388 and attribute_id = 62;
SELECT value into  @destiption  FROM `catalog_product_entity_text` WHERE entity_id = 388 and attribute_id = 61;
update catalog_product_entity_text set value = concat( @short_destiption,@destiption)  where  entity_id = 388 and attribute_id ='61';
COMMIT;

务操作,要保证的三个原则性:

原子性(Atomicity):事务是一个原子操作单元,其对数据的修改,要么全部执行,要么全都不执行;

一致性(Consistent):在事务开始和完成时,数据都必须保持一致状态;

隔离性(Isolation):数据库系统提供一定的隔离机制,保证事务在不受外部并发操作影响的“独立”环境执行;

持久性(Durable):事务完成之后,它对于数据的修改是永久性的,即使出现系统故障也能够保持。

于是,我们假设两个对象A和B

并发对象A 和B

初始状态数据表查询结果:

事务开始的顺序 A->B

A:开始事务

此刻没有提交进行commit

在此状态下B开始了自己的MySQL事务处理:

显然,在A没有进行Commit行为的时候,B的事务中的动作无法完成,一直处于事务等待阶段,前提是在没有超出时限,B的动作无法提交。

此刻,如果A进行Commit:

此刻 B的动作中,由等待的

转变到

说明A完成事务开始解锁,B事务可以进行,但是此刻B事务没有提交(Commit)

注意:在这里我们进行两个操作,就是两个对象进行查询

A的查询:

依然存在ID为1的这项,原因是B的结果没提交,但A依旧可以读该项数据,但是数据为删除前的数据。

但是,对照B的查询:

可以知道,B对象结果已经在处理了,只是没有提交解锁。

分析可以知道,A读的是B没有提交前的结果集合,但B读的是自己操作的结果集,当B完成提交的时候

此刻,A的结果集合

发现现在状态下和B的集合一样,A=B,这也是在理论值的范围内的。

由此,可以发现其实MySQL锁的简单性,当然,也说明当数据库进行锁操作时候,只能是写操作控制,对于读数据,往往只能访问到修改前的数据,这部分数据常常被称为”dirty”或脏数据。在现实中,我们常常是有这样的需求,当A进行写操作时候,期望B不要读数据,是对读行为的控制。

1.1.       基础知识和相关概念

1)全部的表类型都可以使用锁,但是只有InnoDB和BDB才有内置的事务功能。

2)使用begin开始事务,使用commit结束事务,中间可以使用rollback回滚事务。

3)在默认情况下,InnoDB表支持一致读。

SQL标准中定义了4个隔离级别:read uncommited,read commited,repeatable read,serializable。

       read uncommited即脏读,一个事务修改了一行,另一个事务也可以读到该行。

       如果第一个事务执行了回滚,那么第二个事务读取的就是从来没有正式出现过的值。?

       read commited即一致读,试图通过只读取提交的值的方式来解决脏读的问题,

       但是这又引起了不可重复读取的问题。

       一个事务执行一个查询,读取了大量的数据行。在它结束读取之前,另一个事务可能完成了对数据行的更改。当第一个事务试图再次执行同一个查询,服务器就会返回不同的结果。

       repeatable read即可重复读,在一个事务对数据行执行读取或写入操作时锁定了这些数据行。

       但是这种方式又引发了幻想读的问题。

       因为只能锁定读取或写入的行,不能阻止另一个事务插入数据,后期执行同样的查询会产生更多的结果。

       serializable模式中,事务被强制为依次执行。这是SQL标准建议的默认行为。

4)如果多个事务更新了同一行,就可以通过回滚其中一个事务来解除死锁。

5)MySQL允许利用set transaction来设置隔离级别。

6)事务只用于insert和update语句来更新数据表,不能用于对表结构的更改。执行一条更改表结构或begin则会立即提交当前的事务。

7)所有表类型都支持表级锁,但是MyISAM只支持表级锁。

8)有两种类型的表级锁:读锁和写锁。

读锁是共享锁,支持并发读,写操作被锁。

写锁是独占锁,上锁期间其他线程不能读表或写表。

8)如果要支持并发读写,建议采用InnoDB表,因为它是采用行级锁,可以获得更多的更新性能。

9)很多时候,可以通过经验来评估什么样的锁对应用程序更合适,不过通常很难说一个锁比别的更好,这全都要依据应用程序来决定,不同的地方可能需要不同的锁。当前MySQL已经支持 ISAM, MyISAM, MEMORY (HEAP) 类型表的表级锁了,BDB 表支持页级锁,InnoDB 表支持行级锁。

10)MySQL的表级锁都是写锁优先,而且是采用排队机制,这样不会出现死锁的情况。对于 InnoDB 和 BDB 存储引擎来说,是可能产生死锁的。这是因为 InnoDB 会自动捕获行锁,BDB 会在执行 SQL 语句时捕获页锁的,而不是在事务的开始就这么做。
1.2.       不同锁的优缺点及选择

行级锁的优点及选择:

1)在很多线程请求不同记录时减少冲突锁。

2)事务回滚时减少改变数据。

3)使长时间对单独的一行记录加锁成为可能。

 

行级锁的缺点:

1)比页级锁和表级锁消耗更多的内存。

2)当在大量表中使用时,比页级锁和表级锁更慢,因为他需要请求更多的所资源。

3)当需要频繁对大部分数据做 GROUP BY 操作或者需要频繁扫描整个表时,就明显的比其它锁更糟糕。

4)使用更高层的锁的话,就能更方便的支持各种不同的类型应用程序,因为这种锁的开销比行级锁小多了。

5)可以用应用程序级锁来代替行级锁,例如MySQL中的 GET_LOCK() 和 RELEASE_LOCK()。但它们是劝告锁(原文:These are advisory locks),因此只能用于安全可信的应用程序中。

6)对于 InnoDB 和 BDB 表,MySQL只有在指定用 LOCK TABLES 锁表时才使用表级锁。在这两种表中,建议最好不要使用 LOCK TABLES,因为 InnoDB 自动采用行级锁,BDB 用页级锁来保证事务的隔离。

 

表锁的优点及选择:

1)很多操作都是读表。

2)在严格条件的索引上读取和更新,当更新或者删除可以用单独的索引来读取得到时:UPDATE tbl_name SET column=value WHERE unique_key_col=key_value;DELETE FROM tbl_name WHERE unique_key_col=key_value;

3)SELECT 和 INSERT 语句并发的执行,但是只有很少的 UPDATE 和 DELETE 语句。

4)很多的扫描表和对全表的 GROUP BY 操作,但是没有任何写表。

 

表锁的缺点:

1)一个客户端提交了一个需要长时间运行的 SELECT 操作。

2)其他客户端对同一个表提交了 UPDATE 操作,这个客户端就要等到 SELECT 完成了才能开始执行。

3)其他客户端也对同一个表提交了 SELECT 请求。由于 UPDATE的优先级高于 SELECT,所以 SELECT 就会先等到 UPDATE 完成了之后才开始执行,它也在等待第一个 SELECT操作。

 
1.3.       如何避免锁的资源竞争

1)让 SELECT 速度尽量快,这可能需要创建一些摘要表。

2)启动 mysqld 时使用参数 --low-priority-updates。这就会让更新操作的优先级低于 SELECT。

这种情况下,在上面的假设中,第二个 SELECT 就会在 INSERT 之前执行了,而且也无需等待第一个SELECT 了。

3)可以执行 SET LOW_PRIORITY_UPDATES=1 命令,指定所有的更新操作都放到一个指定的链接中去完成。

4)用 LOW_PRIORITY 属性来降低 INSERT,UPDATE,DELETE 的优先级。

5)用HIGH_PRIORITY 来提高 SELECT 语句的优先级。

6)从MySQL 3.23.7 开始,可以在启动 mysqld 时指定系统变量 max_write_lock_count 为一个比较低的值,它能强制临时地提高表的插入数达到一个特定值后的所有 SELECT 操作的优先级。它允许在 WRITE 锁达到一定数量后有 READ 锁。

7)当 INSERT 和 SELECT 一起使用出现问题时,可以转而采用 MyISAM 表,它支持并发的SELECT 和 INSERT 操作。

8)当在同一个表上同时有插入和删除操作时,INSERT DELAYED 可能会很有用。

9)当SELECT 和 DELETE 一起使用出现问题时,DELETE 的 LIMIT 参数可能会很有用。

10)执行SELECT 时使用 SQL_BUFFER_RESULT 有助于减短锁表的持续时间。

11)可以修改源代码 `mysys/thr_lock.c',只用一个所队列。这种情况下,写锁和读锁的优先级就一样了,这对一些应用可能有帮助。

时间: 2024-08-02 08:48:50

php mysql事务处理的相关文章

mysql事务处理用法与实例详解

一个事务是一个连续的一组数据库操作,就好像它是一个单一的工作单元进行.换言之,永远不会是完整的事务,除非该组内的每个单独的操作是成功的.如果在事务的任何操作失败,则整个事务将失败. 实际上,会俱乐部许多SQL查询到一个组中,将执行所有的人都一起作为事务的一部分. 事务的特性: 事务有以下四个标准属性的缩写ACID,通常被称为: 原子性: 确保工作单元内的所有操作都成功完成,否则事务将被中止在故障点,和以前的操作将回滚到以前的状态. 一致性: 确保数据库正确地改变状态后,成功提交的事务. 隔离性:

mysql事务处理学习笔记

什么是事务 在MySQL环境中,一个事务由作为一个单独单元的一个或者多个sql语句组成.这个单元中的每个sql语句是互相依赖的, 而且单元作为一个整体是不可分割的.如果单元中的一个语句不能成功完成,整个单元都会回滚,所有影响到的数据将返回到 事务开始之前的状态.因而,只有事务中的所有语句都被成功的执行才能说这个事务被成功执行. 事务的四个特性: 原子性,每个事务都必须被认为是一个不可分割的单元. 一致性,不管事务是完全成功还是中途失败,当事务使系统处于一致的状态时存在一致性. 孤立性,每个事务在

php下pdo的mysql事务处理用法实例_php技巧

本文实例讲述了php下pdo的mysql事务处理用法.分享给大家供大家参考.具体分析如下: php+mysql事务处理的几个步骤: 1.关闭自动提交 2.开启事务处理 3.有异常就自动抛出异常提示再回滚 4.开启自动提交 注意:mysql只有这个InnoDB驱动是支持事务处理的,默认MyIsAM驱动不支持,下面是实例代码: 复制代码 代码如下: <?php     try{         $pdo=new pdo("mysql:host=localhost;dbname=mydb&quo

MYSQL事务处理之学习笔记

mysql事务处理的几个步骤: 1.关闭自动提交 2.开启事务处理 3.有异常就自动抛出异常提示再回滚 4.开启自动提交 注意:mysql只有这个InnoDB驱动是支持事务处理的,默认MyIsAM驱动不支持. 由于项目设计里面,牵扯到了金钱的转移,于是就要用到MYSQL的事务处理,来保证一组处理结果的正确性.用了事务,就不可避免的要牺牲一部分速度,来保证数据的正确性. 只有InnoDB支持事务 事务 ACID Atomicity(原子性).Consistency(稳定性).Isolation(隔

Mysql事务处理问题

  今天和同学讨论起数据库事务处理的问题,感觉讨论中明白了一些,有些知识看过了,但是没有实际使用还是不理解. 事务处理就是将一系列操作当做一个原子操作,要么全部执行成功,如果执行失败则保留执行期的状态.通过提交和回滚机制来实现操作,如果全部执行成功通过提交执行commit结果就会记录到数据库中,如果执行失败通过回滚操作rollback将发生错误之前的所有错误消除,回退到原来状态. 事务都应该具备ACID特征.所谓ACID是Atomic(原子性),Consistent(一致性),Isolated(

mysql 事务处理及表锁定深入简析_Mysql

MYSQL的事务处理主要有两种方法. 1.用begin,rollback,commit来实现 begin 开始一个事务 rollback 事务回滚 commit 事务确认 2.直接用set来改变mysql的自动提交模式 MYSQL默认是自动提交的,也就是你提交一个QUERY,它就直接执行!我们可以通过 set autocommit=0 禁止自动提交 set autocommit=1 开启自动提交 来实现事务的处理. 当你用 set autocommit=0 的时候,你以后所有的SQL都将做为事务

MySQL事务处理与应用简析_Mysql

事务处理在各种管理系统中都有着广泛的应用,比如人员管理系统,很多同步数据库操作大都需要用到事务处理.比如说,在人员管理系统中,你删除一个人员,你即需要删除人员的基本资料,也要删除和该人员相关的信息,如信箱,文章等等,这样,这些数据库操作语句就构成一个事务!删除的SQL语句delete from userinfo where ~~~delete from mail where ~~delete from article where~~~~如果没有事务处理,在你删除的过程中,假设出错了,只执行了第一

MYSQL事务处理使用说明

MYSQL的事务处理主要有两种方法. 1.用begin,rollback,commit来实现 begin 开始一个事务 rollback 事务回滚 commit 事务确认 2.直接用set来改变mysql的自动提交模式 MYSQL默认是自动提交的,也就是你提交一个QUERY,它就直接执行!我们可以通过 set autocommit=0 禁止自动提交 set autocommit=1 开启自动提交 来实现事务的处理. 当你用 set autocommit=0 的时候,你以后所有的SQL都将做为事务

解析php mysql 事务处理回滚操作(附实例)_php技巧

很多新手在进行项目过程中,会碰到这样一种情况,如:论坛扣币项目中,用户支付论坛币的时候如果突然断网.电脑死机.停电.等其它自然灾害时,导致本次交易没有成功(即用户的币已经扣掉了,但是服务器数据库中没有消费记录等其它情况),这种情况应该怎么样进行处理呢? 在这个时候,我们可以用Mysql事务回滚来进行处理,那代码怎么写呢? 那下面我就来说一说这个mysql事务回滚应该怎么处理. 首先MYSQL中只有INNODB和BDB类型的数据表才能支持事务处理!其他的类型是不支持的! 那如果,我们的数据表已经存