StackExchange.Redis事务方法冻结

时间:2014-09-22 14:12:51

标签: redis stackexchange.redis

我有这个代码在Stackexchange.Redis中添加对象和索引字段。 事务冻结线程中的所有方法。 为什么?

  var transaction = Database.CreateTransaction();

  //this line freeze thread. WHY ?
  await transaction.StringSetAsync(KeyProvider.GetForID(obj.ID), PreSaveObject(obj));
  await transaction.HashSetAsync(emailKey, new[] { new HashEntry(obj.Email, Convert.ToString(obj.ID)) });

  return await transaction.ExecuteAsync();

3 个答案:

答案 0 :(得分:16)

在执行事务之前之后,事务中执行的命令不会返回结果。这只是Redis中交易如何工作的一个特征。目前,您正在等待尚未发送的内容(事务在本地缓存,直到执行) - 但即使已发送:结果根本无法使用直到交易完成。

如果你想要结果,你应该存储(而不是等待)任务,并在执行之后等待它

var fooTask = tran.SomeCommandAsync(...);
if(await tran.ExecuteAsync()) {
    var foo = await fooTask;
}

请注意,这比它看起来要便宜:当事务执行时,嵌套任务会同时获得结果 - 而await可以有效地处理该场景。

答案 1 :(得分:1)

Marc的回答是有效的,但在我的情况下,它导致了相当多的代码膨胀(并且很容易忘记这样做),所以我提出了一种抽象来强制执行模式。

以下是您使用它的方式:

await db.TransactAsync(commands => commands
    .Enqueue(tran => tran.SomeCommandAsync(...))
    .Enqueue(tran => tran.SomeCommandAsync(...))
    .Enqueue(tran => tran.SomeCommandAsync(...)));

以下是实施:

public static class RedisExtensions
{
    public static async Task TransactAsync(this IDatabase db, Action<RedisCommandQueue> addCommands) 
    {
        var tran = db.CreateTransaction();
        var q = new RedisCommandQueue(tran);

        addCommands(q);

        if (await tran.ExecuteAsync())
            await q.CompleteAsync();
    }
}

public class RedisCommandQueue
{
    private readonly ITransaction _tran;
    private readonly IList<Task> _tasks = new List<Task>();

    public RedisCommandQueue Enqueue(Func<ITransaction, Task> cmd)
    {
        _tasks.Add(cmd(_tran));
        return this;
    }

    internal RedisCommandQueue(ITransaction tran) => _tran = tran;
    internal Task CompleteAsync() => Task.WhenAll(_tasks);
}

有一点需要注意:这并不能提供一种简单的方法来获取任何命令的结果。在我的情况下(和OP),没关系 - 我总是使用事务进行一系列的写操作。我发现这确实有助于减少我的代码,并且只在tran内部Enqueue(这要求你返回一个任务),我不太可能“忘记”我不应该在我打电话给他们的时候await这些命令。

答案 2 :(得分:0)

我和我们的团队多次被这个问题困扰,因此我创建了一个简单的Roslyn分析仪来发现此类问题。

https://github.com/olsh/stack-exchange-redis-analyzer