簡體   English   中英

跨 Redis 數據結構原子地執行命令

[英]Atomically Execute commands across Redis Data Structures

我想自動執行一些 redis 命令(HDel、SADD、HSet 等)。 我在 go-redis 中看到了 Watch 功能來實現事務,但是由於我不打算修改鍵的值,即使用 SET、GET 等,使用 Watch 將其作為事務執行或只是包裝是否有意義TxPipeline 中的命令就足夠了嗎?

方法一:使用手表

func sampleTransaction() error{
    transactionFunc := func(tx *redis.Tx) error {
        // Get the current value or zero.

        _, err := tx.TxPipelined(context.Background(), func(pipe redis.Pipeliner) error {
            _, Err := tx.SAdd(context.Background(), "redis-set-key", "value1").Result()
            if Err != nil {
                return Err
            }

            _, deleteErr := tx.HDel(context.Background(), "redis-hash-key", "value1").Result()
            if deleteErr != nil {
                return deleteErr
            }
            return nil
        })
        return err
    }

retries:=10
    // Retry if the key has been changed.
    for i := 0; i < retries; i++ {
        fmt.Println("tries", i)
        err := redisClient.Watch(context.Background(), transactionFunc())
        if err == nil {
            // Success.
            return nil
        }
        if err == redis.TxFailedErr {
            continue
        }
        return err
    }
}

方法 2:僅包裝在 TxPipelined 中

func sampleTransaction() error {
 _, err:= tx.TxPipelined(context.Background(), func(pipe redis.Pipeliner) error {
            _, Err := tx.SAdd(context.Background(), "redis-set-key", "value1").Result()
            if Err != nil {
                return Err
            }

            _, deleteErr := tx.HDel(context.Background(), "redis-hash-key", "value1").Result()
            if deleteErr != nil {
                return deleteErr
            }
            return nil
        })
return err
}

據我所知,管道不保證原子性。 如果需要原子性,使用lua。https://pkg.go.dev/github.com/mediocregopher/radix.v3#NewEvalScript

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM