MySQL中的事务,默认是自动提交的,即autocommit = 1;
但是这样的话,在某些情形中就会出现问题:比如:
如果你想一次性插入了1000条数据,mysql会commit1000次的,
如果我们把autocommit关闭掉[autocommit = 0],通过程序来控制,只要一次commit就可以了,这样也才能更好的体现事务的特点!
 
对于需要操作数值,比如金额,个数等等!
记住一个原则:一锁二判三更新
 
如果SELECT 后面若要UPDATE 同一个表单,最好使用SELECT ... FOR UPDATE
 
举个例子: 假设商品表单products 内有一个存放商品数量的quantity ,在订单成立之前必须先确定quantity 商品数量是否足够(quantity>0) ,
然后才把数量更新为1。
 
不安全的做法:
 
SELECT quantity FROM products WHERE id=3; UPDATE products SET quantity = 1 WHERE id=3;
 
为什么不安全呢?
 
少量的状况下或许不会有问题,但是大量的数据存取「铁定」会出问题。
 
如果我们需要在quantity>0 的情况下才能扣库存,假设程序在第一行SELECT 读到的quantity 是2 ,看起来数字没有错,但是当MySQL 正准备要UPDATE 的时候,可能已经有人把库存扣成0 了,但是程序却浑然不知,将错就错的UPDATE 下去了。
 
因此必须透过的事务机制来确保读取及提交的数据都是正确的。
 
于是我们在MySQL 就可以这样测试(请确保数据库表的引擎是Innodb而不是MyIsAM): 
 
SET AUTOCOMMIT=0; BEGIN WORK; SELECT user_tel FROM users WHERE user_id='60' FOR UPDATE;
=================================================================================
 
此时users数据中id=60 的数据被锁住,其它事务必须等待此次事务 提交后才能执行!
 
另外开一个窗口测试,
1. SELECT * from users where user_id='60' FOR UPDATE;    因为上面事务还没有提交,id =60的数据已经被锁住,此处再执行的话数据库会报错!
[Err] 1205 - Lock wait timeout exceeded; try restarting transaction
 
2.在另外一个窗口中来对表进行更新操作一样不能成功,会返回同样的错误。
 
3.SELECT * FROM usersWHERE id=60     则不会受影响!
 
SELECT * FROM users WHERE id=60 FOR UPDATE    如此可以确保user_tel在别的事务读到的数字是正确的。
 ================================================================================
必须在锁表语句的同一个回话里执行更新操作才能够释放锁
UPDATE users SET user_tels= '111212122' WHERE id=60 ; COMMIT WORK;
 
==================================================================================
 
上面介绍过SELECT ... FOR UPDATE 的用法,不过锁定(Lock)的数据是判别就得要注意一下了。
由于InnoDB[MySQL数据库引擎之一] 预设是Row-Level Lock,所以只有「明确」的指定主键,MySQL 才会执行Row lock (只锁住被选取的数据) ,
否则MySQL 将会执行Table Lock (将整个数据表单给锁住)[严重影响效率]。
举个例子:
 
假设有个表单products ,里面有id 跟name 二个栏位,id 是主键。
 
例1: (明确指定主键,并且有此数据,row lock)
 
SELECT * FROM products WHERE id='3' FOR UPDATE;
 
例2: (明确指定主键,若查无此数据,无lock)
 
SELECT * FROM products WHERE id='-1' FOR UPDATE;
 
例2: (无主键,table lock)
 
SELECT * FROM products WHERE name='Mouse' FOR UPDATE;
 
例3: (主键不明确,table lock)
 
SELECT * FROM products WHERE id<>'3' FOR UPDATE;
 
例4: (主键不明确,table lock)
 
SELECT * FROM products WHERE id LIKE '3' FOR UPDATE;
 
 
注1: FOR UPDATE 仅适用于InnoDB,且必须在事务区块(BEGIN/COMMIT)中才能生效。
 
注2: 要测试锁定的状况,可以利用MySQL 的Command Mode ,开二个视窗来做测试。
 
 
 
以上

【转载】MySQL事务以及SELECT ... FOR UPDATE的使用的更多相关文章

  1. mysql事务,select for update,及数据的一致性处理

    在MySQL的InnoDB中,预设的Tansaction isolation level 为REPEATABLE READ(可重读) 在SELECT 的读取锁定主要分为两种方式: SELECT ... ...

  2. mysql 多列唯一索引在事务中select for update是不是行锁?

    在表中有这么一索引 UNIQUE KEY `customer_id` (`customer_id`,`item_id`,`ref_id`) 问1. 这种多列唯一索引在事务中select for upd ...

  3. 关于mysql事务行锁for update实现写锁的功能

    关于mysql事务行锁for update实现写锁的功能 读后感:用切面编程的理论来讲,数据库的锁对于业务来说是透明的.spring的事务管理代码,业务逻辑代码,表锁,应该是三个不同的设计层面. 在电 ...

  4. mysql的事务和select...for update

    一.mysql的事务mysql的事务有两种方式:1.SET AUTOCOMMIT=0;也就是关闭了自动提交,那么任何commit或rollback语句都可以触发事务提交;如果SET AUTOCOMMI ...

  5. MySQL中的 ”SELECT FOR UPDATE“ 一次实践

    背景 最近工作中遇到一个问题,两个不同的线程会对数据库里的一条数据做修改,如果不加锁的话,会得到错误的结果. 就用了MySQL中for update 这种方式来实现 本文主要测试主键.唯一索引和普通索 ...

  6. MySQL事务实现原理

    MySQL事务隔离级别的实现原理 知识储备 只有InnoDB支持事务,所以这里说的事务隔离级别是指InnoDB下的事务隔离级别 隔离级别 读未提交:一个事务可以读取到另一个事务未提交的修改.这会带来脏 ...

  7. PHP处理mysql事务

    MYSQL的事务处理主要有两种方法.1.用begin,rollback,commit来实现begin 开始一个事务rollback 事务回滚commit 事务确认2.直接用set来改变mysql的自动 ...

  8. PHP操作MySQL事务实例

    PHP与MYSQL事务处理 一般来说,事务都应该具备ACID特征.所谓ACID是Atomic(原子性),Consistent(一致性),Isolated(隔离性),Durable(持续性)四个词的首字 ...

  9. 深度剖析 MySQL 事务隔离

    概述 今天主要分享下MySQL事务隔离级别的实现原理,因为只有InnoDB支持事务,所以这里的事务隔离级别是指InnoDB下的事务隔离级别. 隔离级别 读未提交:一个事务可以读取到另一个事务未提交的修 ...

随机推荐

  1. javac 小记

    javac 到底是什么? javac 就是一个编译器,它把 Java 源代码编译成 Java 字节码,即 JVM 能够识别的二进制形式的文件. javac 由什么构成? 词法分析器:识别源代码中的 J ...

  2. EGOCache缓存框架具体解说

    watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/ ...

  3. POJ 2367 topological_sort

    Genealogical tree Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 2920 Accepted: 1962 Spe ...

  4. Ration Rose2003安装及破解

    曾经学习UML的时候,用的是EA. 近期在看Head First想着画绘图装一下Ration Rose吧.于是就着手開始装.本来网上关于Ration Rose的安装及破解教程非常多,可是我在安装的过程 ...

  5. java反射(转)

    作者:奋斗的小子链接:https://www.zhihu.com/question/24304289/answer/38218810来源:知乎著作权归作者所有,转载请联系作者获得授权. 反射之中包含了 ...

  6. js、jquery实现模糊搜索功能

    模糊搜索功能在工作中应用广泛,并且很实用,自己写了一个方法,以后用到的时候可以直接拿来用了! 实现的搜索功能: 1. 可以匹配输入的字符串找出列表中匹配的项,列表框的高度跟随搜索出的列表项的多少改变 ...

  7. Cenos 6.5上的subverion的yum配置笔记

    Subversion在CenOS 6.5上的安装配置 1.安装 yum install subversion 2.配置   #创建目录   mkdir /opt/svn   #创建版本库   svna ...

  8. UIImage类方法总结及UIImage生成方法对比

    http://mp.weixin.qq.com/s/A900w0Y5pGjuaB4j9Os9ww1.UIImage 生成方法的对比 Apple官方的文档为生成一个UIImage对象提供了两种方法: 1 ...

  9. iOS屏幕适配(尺寸适配)

    屏幕尺寸适配:一 在.pch中加入以下代码,在定义每个尺寸值的时候都调用下边的宏 //以iphone7为例 定义 view相关的宽高宏#define IPHONEHIGHT(b) [UIScreen ...

  10. iOS 如何优化 App 的启动时间

    App 运行理论 main() 执行前发生的事 Mach-O 格式 虚拟内存基础 Mach-O 二进制的加载 理论速成 Mach-O 术语 Mach-O 是针对不同运行时可执行文件的文件类型. 文件类 ...