Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,提供多种语言的API,支持多种数据存储方式。开发中一般用于缓存,队列,数据库使用。

redis事务(Transaction)说明

redis中的事务与传统关系型数据库(如mysql)的事务是不同的。

redis中的事务是一组命令的集合,事务与命令都是最小执行单位,原理是先将属于一个事务的命令发送给Redis,然后Redis一次执行这些命令。

redis的事务可以保证一个事务内的命令一次执行而不被其他命令插入影响。

redis的事务错误处理与关系型数据库事务错误处理区别

如果事务块中某一条命令出错,关系型数据库的事务会执行回滚,而redis不会执行回滚,而是会继续执行后续的命令。因为redis的事务没有关系型数据库的回滚(rollback)功能。因此需要开发者在事务执行出错时自己处理。

redis事务(Transaction)命令

1.watch
用于监视一个或多个key,如果在事务执行之前这个或(这些)key被其他命令所改动,事务将被中断。

2.unwatch
用于取消watch命令对所有key的监视。

3.multi
用于标记一个事务块的开始,之后的所有命令都存放在队列,等遇到exec命令再执行。

4.exec
用于执行事务块内所有的命令,如果命令被中断,返回false

实例

1.使用redis事务,事务正常执行

<?php
$redis = new Redis;
$redis->connect('localhost', 6379);

// 监听
$redis->watch('mykey1','mykey2');

// 开启事务块
$redis->multi();

// 事务块语句
$redis->set('mykey1', date('Y-m-d H:i:s'));
$redis->set('mykey2', time());

// 执行事务
$result = $redis->exec();

// 取消监听
$redis->unwatch();

var_dump($result);
?>

输出:

array (size=2)
  0 => boolean true
  1 => boolean true

进入redis查看这两个key的值

127.0.0.1:6379> mget "mykey1" "mykey2"
1) "2017-09-24 17:14:02"
2) "1506244442"

表示事务块语句 set mykey1,mykey2执行成功。

2.使用redis事务,事务被中断

我们模拟在事务执行过程中值被修改,因此在事务块中加入sleep延迟执行,然后在延迟的时候修改监听key的值。

<?php
$redis = new Redis;
$redis->connect('localhost', 6379);

// 监听
$redis->watch('mykey1','mykey2');

// 开启事务块
$redis->multi();

// 事务块语句
$redis->set('mykey1', date('Y-m-d H:i:s'));
$redis->set('mykey2', time());

// 延迟5秒
sleep(5);

// 执行事务
$result = $redis->exec();

// 取消监听
$redis->unwatch();

var_dump($result);
?>

在延迟的时候,进入redis,手动执行以下命令修改mykey2的值。

127.0.0.1:6379> set mykey2 123
OK

因为监听的key的值被修改,因此事务执行中止。

输出:

boolean false

进入redis查看这两个key的值

127.0.0.1:6379> mget "mykey1" "mykey2"
1) "2017-09-24 17:14:02"
2) "123"

表示事务块语句 set mykey1,mykey2没有执行。