我有这个代码在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();
答案 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分析仪来发现此类问题。