异步函数进入死锁

本文关键字:死锁 函数 异步 | 更新日期: 2023-09-27 18:06:44

目标:

我正在使用redis进行缓存,并尝试获取缓存值(如果不可用(,请从repo获取并设置缓存值。

问题:

客户端js一直在等待,它似乎处于某种类型的死锁中

导致问题的代码

   return await _cacheStore.GetAsync("CacheKey", new TimeSpan(24, 0, 0),
                 async () =>
                  await _repo.GetAllAsync()
                  );

代码有效,但有点多余,因为我在很多地方一直使用这种模式

    var data = await _cacheStore.GetAsync<List<objectType>>("cacheKey");
    if (data == null)
    {
        data = await _repo.GetAllAsync();
        _cacheStore.SetAsync<List<objectType>>("cacheKey", data, new TimeSpan(24, 0, 0));
    }
    return data;

//将异步方法传递为Func下面可能存在的问题函数?????以上代码调用的函数

public static async Task < T > GetAsync < T > (this IRedisCacheStore source, string key, TimeSpan time, Func < Task < T >> fetch) where T: class {
 if (source.Exists(key)) {
  return await source.GetAsync < T > (key);
 } else {
  var result = fetch();
  if (result != null) {
   source.Set(key, result, time);
  }
  return await result;
 }
}
public async Task < List < ObjectType >> GetAllAsync() {
 var result = await _procExecutor.ExecuteProcAsync < ObjectType > ("Sproc", null);
 return (result ? ? new List < ObjectType > ()).ToList();
}
public async Task < IEnumerable < TResult >> ExecuteProcAsync < TResult > (string sproc, object param) {
  using(var conn = _connection.GetConnection()) {
   return await conn.QueryAsync < TResult > (sproc, param, commandType: CommandType.StoredProcedure);
  }

这是我的redis cacheStore

public interface IRedisCacheStore
{
    bool Exists(string key);
    T Get<T>(string key);
    void Set<T>(string key, T value, TimeSpan expiredIn);
    void Remove(string key);
    Task<bool> ExistsAsync(string key);
    Task<T> GetAsync<T>(string key);
    Task SetAsync<T>(string key, T value, TimeSpan expiredIn);
    Task RemoveAsync(string key);
}

异步函数进入死锁

问题就在这里:

var result = fetch();
if (result != null) {
   source.Set(key, result, time);
} 

fetch()返回Task<T>,这就是您试图存储在Redis缓存中的内容,而Task(显然(是不可序列化的。只需使用:

var result = await fetch();