更新时间:2025-04-24 GMT+08:00
GeminiDB Redis事务处理
事务介绍
事务将应用程序的多个读、写操作捆绑在一起成为一个逻辑操作单元,是一个执行的整体,整个事务要么成功,要么失败。在一个连接中,当客户端执行multi命令后,redis开始将后续收到的命令缓存在队列中,当客户端发送exec命令时,redis按照顺序依次执行队列中的所有命令。如果有一个命令执行失败则事务回滚,所有命令要么全部成功,要么全部失败。
|
命令 |
含义 |
|---|---|
|
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();
}
}