Redis只支持简单的事务,不像mysql那样比较完整严格,对数据的完整性也维持的很好。redis的开启事务实际上只是将开启事务之后的一段命令用队列
包裹起来了,当调用redis的执行命令(exec)全部执行在队列中的命令。
一、mysql事务与redis事务的比较

 

Mysql

Redis

开启

start transaction

multi

语句

普通sql

普通命令

失败

rollback 回滚

discard 取消

成功

commit

exec

二、redis事务一般使用演示
(1).先清空所有数据
127.0.0.1:6379> flushdb
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set t1 "test1 value"
QUEUED
127.0.0.1:6379> set t2 "test2 value"
QUEUED
127.0.0.1:6379>
(2).当没有执行提交的时,打开另外一个redis命令窗口执行keys *发现并没有新的数据产生
127.0.0.1:6379> keys *
(empty list or set)
(3).在窗口一种执行提交
127.0.0.1:6379> exec
1) OK
2) OK
(4).另一窗口显示所有key
127.0.0.1:6379> keys *
1) "t1"
2) "t2"

三、redis事务若是出错并不回滚
redis事务命令列中如果出错,整个事务并不回滚,而是会继续执行下去,所以这就违反了事务中ACID特点的原子性
redis事务出错的两种情况:
(1).redis语法有问题导致出错
执行exec后的结果:所有multi后执行的命令都不会被执行

演示如下:
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set a a
QUEUED
127.0.0.1:6379> set b
(error) ERR wrong number of arguments for 'set' command
127.0.0.1:6379> exec
(error) EXECABORT Transaction discarded because of previous errors.
127.0.0.1:6379>
查看最终结果key:
127.0.0.1:6379> keys *
1) "t1"
2) "t2"
总结:并没有找到multi后执行的命令结果key,说明全部失败了

(2).redis语法没问题但是操作的适用对象有问题,比如sadd(设置set类型的值)操作string类型的key
执行exec的结果:除了redis语法操作的命令不被正确执行,其它不会被影响正常执行(这个是属于程序级别的出错,一般是由程序员控制的)
演示如下:
127.0.0.1:6379> flushdb
OK
127.0.0.1:6379> set num1 3
OK
127.0.0.1:6379> set num2 4
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> incrby num1 3
QUEUED
127.0.0.1:6379> sadd num2 2
QUEUED
127.0.0.1:6379> exec
1) (integer) 6
2) (error) WRONGTYPE Operation against a key holding the wrong kind of value
查看最终结果:
127.0.0.1:6379> mget num1 num2
1) "6"
2) "4"
127.0.0.1:6379>
总结:因为num1,num2都是字符串类型,multi后用sadd操作num2类型错误,而最终执行后只是num2执行失败,num2正常的由数字3增加了3变成了六。

四、redis的锁机制
悲观锁:如果我去买票,我会想世界不和谐,肯定有人跟我抢,我先把票锁起来,只有我能够操作,这就是悲观锁
乐观锁:如果我去买票,哎呀没那么多人买票,还有好多的票了,所以不用上锁,我只需要关注最后剩下多少张票就行了,这就是乐观锁
演示如下
127.0.0.1:6379> keys *
(empty list or set)
如果售票点有1张票,自己手上还有100块钱
127.0.0.1:6379> set ticket 1
OK
127.0.0.1:6379> set money 100
OK
127.0.0.1:6379> mget ticket money
1) "1"
2) "100"
我们先给ticket(票)加锁
127.0.0.1:6379> watch ticket
OK
开始执行事务
127.0.0.1:6379> multi
OK
这个时候火车站只有一张余票了,我发起了两张票的请求
127.0.0.1:6379> decrby ticket 2
QUEUED
扣除80块的票钱
127.0.0.1:6379> decrby money 80
QUEUED
开始执行提交
127.0.0.1:6379> exec
报错了
(error) EXECABORT Transaction discarded because of previous errors.
查看结果没有变化因为票只有一张不够所以整个事务保持原有的结果
127.0.0.1:6379> mget ticket money
1) "1"
2) "100"
unwatch取消所有锁监听
127.0.0.1:6379> unwatch
OK
总结:redis的事务中,启用的是乐观锁,只负责事务中key没有被改动。

Redis事务及锁应用的更多相关文章

  1. redis事务,分布式锁

    事务:一组命令集合 主要命令multi 和exec multi set a 1 sadd s1 a ...... exec 错误处理 (1)语法错误 127.0.0.1:6379> multi ...

  2. redis 事务(悲观锁和乐观锁)

    MULTI 开启事务,后续的命令会被加入到同一个事务中 事务中的操作会发送给客服端,但是不会立即执行,而是将操作放到了该事务对应的一个队列中,服务端返回QUEQUD EXEC 执行EXEC后,事务中的 ...

  3. 分布式缓存技术redis学习系列(三)——redis高级应用(主从、事务与锁、持久化)

    上文<详细讲解redis数据结构(内存模型)以及常用命令>介绍了redis的数据类型以及常用命令,本文我们来学习下redis的一些高级特性. 安全性设置 设置客户端操作秘密 redis安装 ...

  4. 分布式缓存技术redis学习(三)——redis高级应用(主从、事务与锁、持久化)

    上文<详细讲解redis数据结构(内存模型)以及常用命令>介绍了redis的数据类型以及常用命令,本文我们来学习下redis的一些高级特性.目录如下: 安全性设置 设置客户端操作秘密 客户 ...

  5. Redis事务和分布式锁

    Redis事务 Redis中的事务(transaction)是一组命令的集合.事务同命令一样都是Redis最小的执行单位,一个事务中的命令要么都执行,要么都不执行.Redis事务的实现需要用到 MUL ...

  6. 分布式缓存技术redis系列(三)——redis高级应用(主从、事务与锁、持久化)

    上文<详细讲解redis数据结构(内存模型)以及常用命令>介绍了redis的数据类型以及常用命令,本文我们来学习下redis的一些高级特性. 安全性设置 设置客户端操作秘密 redis安装 ...

  7. 一篇和Redis有关的锁和事务的文章

    部分参考链接 Transaction StackExchange.Redis Transaction hashest 正文 Redis 是一种基于内存的单线程数据库.意味着所有的命令是一个接一个的执行 ...

  8. redis 事务 & 锁

    参考:https://www.cnblogs.com/DeepInThought/p/10720132.html Redis不保证原子性:Redis中,单条命令是原子性执行的,但事务不保证原子性,且没 ...

  9. redis事务机制和分布式锁

    Redis事务机制 严格意义来讲,Redis的事务和我们理解的传统数据库(如mysql)的事务是不一样的:Redis的事务实质上是命令的集合,在一个事务中要么所有命令都被执行,要么所有事物都不执行.  ...

随机推荐

  1. MySQL修改字符集编码

    通过修改字符集编码为utf8,彻底解决中文问题. 一. 登录MySQL查看用SHOW VARIABLES LIKE 'character%':下字符集,显示如下: +----------------- ...

  2. 【cf 483 div2 -C】Finite or not?(数论)

    链接:http://codeforces.com/contest/984/problem/C 题意 三个数p, q, b, 求p/q在b进制下小数点后是否是有限位. 思路 题意转化为是否q|p*b^x ...

  3. Linux下的ASLR(PIE)内存保护机制

    1.1    Linux下的ASLR内存保护机制 1.1.1    Linux下的ASLR工作原理 工作原理与window下的aslr类似 1.1.2 Linux下利用内存地址泄露绕过ASLR ⑴.  ...

  4. qml 音乐播放器的进度条

    进度条采用qml的Slider组件 样式什么的,网上很多.我就不列举了.接下来主要说明,进度条是怎样按秒移动的. Slider { id: control    value: 0 stepSize: ...

  5. iOS跳转支付宝付款码和扫一扫页面

    iOS跳转支付宝付款码和扫一扫页面 // 是否支持支付宝 NSURL * myURL_APP_A = [NSURL URLWithString:@"alipay://"]; if ...

  6. 《Effective C++》第5章 实现-读书笔记

    章节回顾: <Effective C++>第1章 让自己习惯C++-读书笔记 <Effective C++>第2章 构造/析构/赋值运算(1)-读书笔记 <Effecti ...

  7. Axure RP 7.0 注册码

    2016.01.25安装可用   Axure RP 7.0 注册码  用户名:axureuser  序列号:8wFfIX7a8hHq6yAy6T8zCz5R0NBKeVxo9IKu+kgKh79FL6 ...

  8. asp页面快速找到菜单按钮转向的页面的方法

    asp页面快速找到菜单按钮转向的页面的方法: 鼠标放在按钮上,右键属性即可查看

  9. CUDA Samples: Long Vector Add

    以下CUDA sample是分别用C++和CUDA实现的两个非常大的向量相加操作,并对其中使用到的CUDA函数进行了解说,各个文件内容如下: common.hpp: #ifndef FBC_CUDA_ ...

  10. 错误:XMLHttpRequest cannot load

    原因:Chrome浏览器不支持本地ajax访问,具体就是ajax不能访问file 有3种解决办法:http://frabbit2013.blog.51cto.com/1067958/1254285 其 ...