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. slurm任务调度系统部署和测试(一)

    1.概述 本博客通过VMware workstation创建了虚拟机console,然后在console内部创建了8台kvm虚拟机,使用这8台虚拟机作为集群,来部署配置和测试slurm任务调度系统. ...

  2. Android音频: 怎样使用AudioTrack播放一个WAV格式文件?

    翻译 By Long Luo 原文链接:Android Audio: Play a WAV file on an AudioTrack 译者注: 1. 因为这是技术文章,所以有些词句使用原文,表达更准 ...

  3. 青否云 - 小程序待办事项 wxapp开源系统

    青否云最新开源系统:小程序待办事项 wxapp-demo 青否云 小程序 demo 下载地址:https://github.com/qingful/wxapp-demo 官网 http://cloud ...

  4. 为什么还坚持.NET? 找一门适合自己的语言去做编程

    为什么还坚持.NET? 找一门适合自己的语言去做编程 接触了.NET快十二年了,现在专注于分布式服务的开发. 中间经历过各种编程语言的诱惑,ios等. 前几年才对自己有比较明确的定位 技术上:找到适合 ...

  5. 【java】打印一个对象即打印出该对象toString()返回值

    public class TestToString { public static void main(String[] args){ Node node1=new Node("东邪&quo ...

  6. Ant学习笔记

    前言:这段时间在学习Ant,发现这是一个很强大的构建工具.你可能使用了很长一段时间,才发现Ant能做数不完的事.总之,个人觉得,Ant学习门槛低,入门简单,能大概看懂程序,写一些简单的脚本即可,剩下在 ...

  7. Linux 学习记录 三(Vim 文书编辑器).

            所有的Unix Like系统都会内建vi文书编辑器,其他的文书编辑器不一定存在,vim是vi的升级版,具有程序编辑的能力,可以主动的以字体颜色辨别语法的正确性,方便程序设计.vim 里 ...

  8. MySQL 如何存储长度较大的varchar与blob

    本文同时发表在https://github.com/zhangyachen/zhangyachen.github.io/issues/96 最近,在工作中遇到了MySQL中如何存储长度较长的字段类型问 ...

  9. [Upper case conversion ] 每个单词的首小写字母转换为对应的大写字母

    Given a string , write a program to title case every first letter of words in string. Input:The firs ...

  10. C#设计模式之二十二备忘录模式(Memeto Pattern)【行为型】

    一.引言   今天我们开始讲"行为型"设计模式的第十个模式,该模式是[备忘录模式],英文名称是:Memento Pattern.按老规矩,先从名称上来看看这个模式,个人的最初理解就 ...