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. VMware虚拟机创建安装之后不出现VMnet1和VMnet8虚拟网卡

    大家可能遇到过安装虚拟机之后,不出现这两张虚拟网卡,造成一系列的网络问题 VMware虚拟机无法将网络改为桥接状态 本人亲试可行的解决办法 首先把你之前安装的VMware虚拟机卸载,清理得一干二净: ...

  2. 用Qt写了一个qq客户端,采用webqq协议,发出来和大家分享一下---大神请无视

    首先做以下声明: 本程序基于腾讯公司的webqq协议开发,所有相关版权归腾讯公司所有.此程序只用于技术交流和学习,不得用于其他方面. ---开发者:雨后星辰,转载请注明出处:http://www.cn ...

  3. Git 的origin和master分析(转)

    转:http://lishicongli.blog.163.com/blog/static/1468259020132125247302/ 首先要明确一点,对git的操作是围绕3个大的步骤来展开的(其 ...

  4. Ubuntu下配置舒服的Python开发环境

    Ubuntu 提供了一个良好的 Python 开发环境,但如果想使我们的开发效率最大化,还需要进行很多定制化的安装和配置.下面的是我们团队开发人员推荐的一个安装和配置步骤,基于 Ubuntu 12.0 ...

  5. FreeMarker初探--安装FreeMarker

    这里安装FreeMarker相当简单,不需要真正的安装过程.仅仅是拷贝 lib/freemarker.jar 到你 Java 应用程序的路径中,让类加载器可以发现它.比如,如果你在 Web 使用了 F ...

  6. 快速切题 poj 3026 Borg Maze 最小生成树+bfs prim算法 难度:0

    Borg Maze Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 8905   Accepted: 2969 Descrip ...

  7. APUE学习笔记——8.11 实际用户ID、有效用户ID、设置用户ID

    用户ID的基本概念 在Unix系统中,很多操作涉及到权限问题,这些权限涉及到用户ID和组ID的概念.     组ID和用户ID的原理和相关内容是类似的.下面介绍用户ID.     我们常见见到三种关于 ...

  8. New Concept English three(21)

    27W 59 Boxing matches were very popular in England two hundred years ago. In those days, boxers foug ...

  9. 两个init方法的区别

    容器创建了Servlet实例后,它将调用实例的init(ServletConfig)方法初始化Servlet.该方法的参数ServletConfig对象包含了在WEB应用程序的部署描述文件中指定的初始 ...

  10. Shell 命令行,实现对若干网站状态批量查询是否正常的脚本

    Shell 命令行,实现对若干网站状态批量查询是否正常的脚本 如果你有比较多的网站,这些网站的运行状态是否正常则是一件需要关心的事情.但是逐一打开检查那简直是一件太糟心的事情了.所以,我想写一个 sh ...