更新时间:2024-12-02 GMT+08:00

GeminiDB Redis事务处理

事务介绍

事务将应用程序的多个读、写操作捆绑在一起成为一个逻辑操作单元,是一个执行的整体,整个事务要么成功,要么失败。在一个连接中,当客户端执行multi命令后,redis开始将后续收到的命令缓存在队列中,当客户端发送exec命令时,redis按照顺序依次执行队列中的所有命令。如果有一个命令执行失败则事务回滚,所有命令要么全部成功,要么全部失败。

表1 相关命令

命令

含义

WATCH

用于监视一个或多个key,如果事务执行之前该key被改动,则事务被打断。

UNWATCH

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

MULTI

用于标识一个事务块的开始。

EXEC

用于执行事务块内的所有命令。

DISCARD

用于取消事务块,放弃执行事务块内的所有命令。

  • 使用proxy集群时,为保证事务执行的原子性,需确保事务中的所有key拥有相同的hashtag。若不使用hashtag,事务会被拆分成普通命令执行,则无法保证事务的原子性。
  • 由于整个事务的执行是原子性的,要么全部成功要么全部失败,因此,在编写事务时,需要注意命令的合法性。
  • 由于事务中的命令是按照顺序执行的,所以,在编写事务时,需要注意命令的先后顺序。
  • 使用事务时应遵循轻量化原则,避免单次事务中打包过多命令或复杂度过高的命令,执行的事务中包含命令过多会引起请求阻塞或实例状态异常。

代码示例

Client在事务中同时修改key1和key2,key1和key2要么同时修改成功,要么同时修改失败。

package nosql.cloud.huawei.jedis;
import java.util.List;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.Transaction;

public class TranscationTest {
    private static final String host = "127.0.0.1";
    private static final int port = 6379;
    private static final String pwd = "password";

    private static Jedis jedis;
    static {
        jedis = new Jedis(host, port);
        String authString = jedis.auth(password);
        if (!authString.equals("OK")) {
            jedis.close();
            jedis = null;
        }
    }
    public static void main(String[] args) {
        if (jedis == null) {
            return;
        }

        String str_key1 = "{str}key1";
        String str_key2 = "{str}key2";
        jedis.set(str_key1, "0");
        jedis.set(str_key2, "0");
        jedis.watch(str_key1);
        // 启动事务处理
        Transaction tx = jedis.multi();
        tx.set(str_key1, "500");
        tx.get(str_key1);
	tx.set(str_key2, "1000");
        tx.get(str_key2);
        List<Object> result = tx.exec();
        if (result.isEmpty()) {
            System.out.println("Error: 事务执行中断");
        } else {
		   System.out.println("Succ: 事务执行成功");
        }
        System.out.println("str_key1: {}, str_key2: {}", jedis.get(str_key1), jedis.get(str_key2));
        jedis.close();
    }
}