Redis事务详述

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: Redis类似大多数成熟的数据库系统一样,提供了事务机制。Redis的事务机制非常简单,它没有严格的事务模型,无法像关系型数据库一样保证操作的原子性。

1、简介

Redis类似大多数成熟的数据库系统一样,提供了事务机制。Redis的事务机制非常简单,它没有严格的事务模型,无法像关系型数据库一样保证操作的原子性。

Redis事务最大的作用是保证多个指令的串行执行,它可以借助于Redis单线程读写的特性,保证Redis事务中的指令不会被事务外的指令打搅,不过要注意它不是原子性的

完整事务案例:

multi开启一个事务之后,所有指令都不执行,而是缓存到事务队列中,直到服务器接收到exec指令,才开始执行整个事务中的指令。事务全部指令执行完毕后,一次性返回全部的结果。

使用Redis事务,一个最需要注意的问题是,指令多,网络开销高;因此我们一定要结合管道pipeline一起使用,这样可以将多次网络io操作压缩成单次。

2、指令介绍

2.1 简介

Redis事务相关的指令有五个,分别是MULTI、EXEC、DISCARD、WATCH、UNWATCH

指令

指令作用

返回值

MULTI

标记一个事务块的开始

总是返回 OK

EXEC

执行所有事务块内的命令

事务块内所有命令的返回值,按命令执行的先后顺序排列。当操作被打断时,返回空值 nil

DISCARD

取消事务,放弃执行事务块内的所有命令,如果正在使用 WATCH 命令监视某个(或某些) key,那么取消所有监视,等同于执行命令 UNWATCH

总是返回 OK

WATCH

监视一个(或多个) key ,如果在事务执行之前这个(或这些) key 被其他命令所改动,那么事务将被打断

总是返回 OK

UNWATCH

取消 WATCH 命令对所有 key 的监视。如果在执行WATCH 命令之后, EXEC 命令或 DISCARD 命令先被执行了的话,那么就不需要再执行 UNWATCH 了。因为 EXEC 命令会执行事务,因此 WATCH 命令的效果已经产生了;而 DISCARD 命令在取消事务的同时也会取消所有对 key 的监视,因此这两个命令执行之后,就没有必要执行 UNWATCH 了

总是返回 OK

2.2 MULTI(开启事务)

MULTI用于标记一个事务的开始,事务块内的多条命令会按照先后顺序被放进一个队列当中,最后由 EXEC 命令原子性(atomic)地执行。MULTI指令总是返回OK。

2.3 EXEC(执行事务)

EXEC用于执行所有事务块内的命令,假如某个(或某些) key 正处于 WATCH 命令的监视之下,且事务块中有和这个(或这些) key 相关的命令,那么 EXEC 命令只在这个(或这些) key 没有被其他命令所改动的情况下执行并生效,否则该事务被打断(abort)。

2.4 DISCARD(取消事务)

DISCARD用于取消事务,放弃执行事务块内的所有命令。如果正在使用 WATCH 命令监视某个(或某些) key,那么取消所有监视,等同于执行命令 UNWATCH 。DISCARD指令总是返回OK。

2.5 WATCH(监视)

WATCH用于监视一个(或多个) key ,如果在事务执行之前这个(或这些) key 被其他命令所改动,那么事务将被打断。这个实现方式也很简单,WATCH是在事务之间发送的指令,Redis服务在接收到指令时,会记录下该key对应的值,当Redis服务接收到EXEC指令,需要执行事务时,Redis服务首先会检查WATCH的key的值,从WATCH之后是否发生改变即可。


注意禁止在MULTI和EXEC之间执行WATCH指令,这会导致Redis服务响应异常


2.6 UNWATCH

UNWATCH用于取消WATCH命令对所有key的监视。如果在执行 WATCH 命令之后, EXEC 命令或 DISCARD 命令先被执行了的话,那么就不需要再执行 UNWATCH 了。因为 EXEC 命令会执行事务,因此 WATCH 命令的效果已经产生了;而 DISCARD 命令在取消事务的同时也会取消所有对 key 的监视,因此这两个命令执行之后,就没有必要执行 UNWATCH 了。


3、Jedis 使用事务

通过模拟一个简单的余额增加的例子,使用Jedis客户端来使用Redis的事务。

package com.lizba.redis.tx;


import redis.clients.jedis.Jedis;

import redis.clients.jedis.Transaction;


import java.math.BigDecimal;

import java.util.List;


/**

* <p>

*      Redis事务demo

* </p>

*

* @Author: Liziba

* @Date: 2021/9/9 23:53

*/

public class TransactionDemo {


   private Jedis client;


   public TransactionDemo(Jedis client) {

       this.client = client;

   }


   /**

    * 添加余额

    *

    * @param userId    用户id

    * @param amt       添加余额

    * @return

    */

   public BigDecimal addBalance(String userId, BigDecimal amt) {

       String key = this.keyFormat(userId);

       // 初始用户余额为0

       client.setnx(key, "0");

       while (true) {

           client.watch(key);

           BigDecimal balance = new BigDecimal(client.get(key)).setScale(2, BigDecimal.ROUND_HALF_UP);

           BigDecimal amount = balance.add(amt);

           Transaction tx = client.multi();

           tx.set(key, amount.toPlainString());

           List<Object> exec = tx.exec();

           // 返回值不为空则证明Redis事务成功

           if (exec != null) {

               break;

           }

       }

       return new BigDecimal(client.get(key)).setScale(2, BigDecimal.ROUND_HALF_UP);

   }


   /**

    * 获取总金额

    *

    * @param userId 用户id

    * @return

    */

   public BigDecimal getAmount(String userId) {

       String amt = client.get(keyFormat(userId));

       return new BigDecimal(amt);

   }


   /**

    * Redis key

    * @param userId 用户id

    * @return

    */

   private String keyFormat(String userId) {

       return String.format("balance:%s",userId);

   }


}


测试代码:

package com.lizba.redis.tx;


import redis.clients.jedis.Jedis;


import java.math.BigDecimal;

import java.util.concurrent.CountDownLatch;


/**

* <p>

*      测试Redis事务

* </p>

*

* @Author: Liziba

* @Date: 2021/9/10 0:03

*/

public class TestTransactionDemo {


   private static CountDownLatch count = new CountDownLatch(100);


   public static void main(String[] args) throws InterruptedException {


       for (int i = 0; i < 100; i++) {

           new Thread(() -> {

               Jedis client = new Jedis("192.168.211.109", 6379);

               TransactionDemo demo = new TransactionDemo(client);

               demo.addBalance("liziba", BigDecimal.TEN);

               client.close();

               count.countDown();

           }).start();

       }


       count.await();


       Jedis client = new Jedis("192.168.211.109", 6379);

       BigDecimal amt = new TransactionDemo(client).getAmount("liziba");

       System.out.println(amt.toPlainString());

   }


}

测试结果:

预期1000,结果1000

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
打赏
0
0
0
0
10
分享
相关文章
|
10月前
|
10- 你们用过Redis的事务吗 ? 事务的命令有哪些 ?
```markdown Redis事务包括MULTI、EXEC、DISCARD、WATCH四个命令。虽具备事务功能,但在实际开发中使用较少。 ```
89 7
什么是Redis的事务?
Redis事务提供原子性和顺序性,确保命令按顺序执行且不被打断。核心概念包括原子性、顺序性、隔离性和持久性。关键指令有MULTI、EXEC、DISCARD和WATCH,用于事务的开始、执行、取消和监视。这保障了命令的完整性,防止并发操作导致的数据不一致。
83 2
Redis事务悄然而至:命令的背后故事
Redis事务悄然而至:命令的背后故事
71 0
|
5月前
|
Redis 事务
10月更文挑战第18天
54 1
|
10月前
|
Redis事务:保证数据操作的一致性和可靠性
Redis事务:保证数据操作的一致性和可靠性
143 0
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
这篇文章是关于Java面试中Redis相关问题的笔记,包括Redis事务实现、集群方案、主从复制原理、CAP和BASE理论以及负载均衡算法和类型。
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
|
3月前
|
Redis事务长什么样?一文带你全面了解
Redis事务是一组命令的有序队列,通过MULTI、EXEC、WATCH和DISCARD等命令实现原子性操作。事务中的命令在EXEC执行前不会实际运行,而是先进入队列,确保所有命令要么全部成功,要么全部失败。此外,Redis还支持Lua脚本实现类似事务的操作,通常更简单高效。事务适用于购物车结算、秒杀活动、排行榜更新等需要保证数据一致性的场景。
57 0
大数据-42 Redis 功能扩展 发布/订阅模式 事务相关的内容 Redis弱事务
大数据-42 Redis 功能扩展 发布/订阅模式 事务相关的内容 Redis弱事务
54 2
Redis 事务特性、原理、具体命令操作全方位诠释 —— 零基础可学习
本文全面阐述了Redis事务的特性、原理、具体命令操作,指出Redis事务具有原子性但不保证一致性、持久性和隔离性,并解释了Redis事务的适用场景和WATCH命令的乐观锁机制。
693 0
Redis 事务特性、原理、具体命令操作全方位诠释 —— 零基础可学习
Redis6入门到实战------ 九、10. Redis_事务_锁机制_秒杀
这篇文章深入探讨了Redis事务的概念、命令使用、错误处理机制以及乐观锁和悲观锁的应用,并通过WATCH/UNWATCH命令展示了事务中的锁机制。
Redis6入门到实战------ 九、10. Redis_事务_锁机制_秒杀
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等