Doing locking in ASP.NET correctly

后端 未结 3 707
广开言路
广开言路 2020-12-29 14:40

I have an ASP.NET site with a fairly slow search function, and I want to improve performance by adding the results to the cache for one hour using the query as the cache-key

相关标签:
3条回答
  • 2020-12-29 15:08

    Unless you're absolutely certain that it's critical to have no redundant queries then I would avoid locking altogether. The ASP.NET cache is inherently thread-safe, so the only drawback to the following code is that you might temporarily see a few redundant queries racing each other when their associated cache entry expires:

    public static string DoSearch(string query)
    {
        var results = (string)HttpContext.Current.Cache[query];
        if (results == null)
        {
            results = GetResultsFromSlowDb(query);
    
            HttpContext.Current.Cache.Insert(query, results, null,
                DateTime.Now.AddHours(1), Cache.NoSlidingExpiration);
        }
        return results;
    }
    

    If you decide that you really must avoid all redundant queries then you could use a set of more granular locks, one lock per query:

    public static string DoSearch(string query)
    {
        var results = (string)HttpContext.Current.Cache[query];
        if (results == null)
        {
            object miniLock = _miniLocks.GetOrAdd(query, k => new object());
            lock (miniLock)
            {
                results = (string)HttpContext.Current.Cache[query];
                if (results == null)
                {
                    results = GetResultsFromSlowDb(query);
    
                    HttpContext.Current.Cache.Insert(query, results, null,
                        DateTime.Now.AddHours(1), Cache.NoSlidingExpiration);
                }
    
                object temp;
                if (_miniLocks.TryGetValue(query, out temp) && (temp == miniLock))
                    _miniLocks.TryRemove(query);
            }
        }
        return results;
    }
    
    private static readonly ConcurrentDictionary<string, object> _miniLocks =
                                      new ConcurrentDictionary<string, object>();
    
    0 讨论(0)
  • 2020-12-29 15:17

    Your code is correct. You are also using double-if-sandwitching-lock which will prevent race conditions which is a common pitfall when not used. This will no lock access to existing stuff in the cache.

    The only problem is when many clients are inserting into the cache at the same time, and they will queue behind the lock but what I would do is to put the results = GetResultsFromSlowDb(query); outside the lock:

    public static string DoSearch(string query)
    {
        string results = "";
    
        if (HttpContext.Current.Cache[query] == null)
        {
            results = GetResultsFromSlowDb(query); // HERE
            lock (_cacheLock)
            {
                if (HttpContext.Current.Cache[query] == null)
                {
    
    
                    HttpContext.Current.Cache.Add(query, results, null, DateTime.Now.AddHours(1), Cache.NoSlidingExpiration, CacheItemPriority.Normal, null);
                }
                else
                {
                    results = HttpContext.Current.Cache[query].ToString();
                }
            }
        }
        else
        {
            results = HttpContext.Current.Cache[query].ToString();
        }
    

    If this is slow, your problem is elsewhere.

    0 讨论(0)
  • 2020-12-29 15:26

    Your code has a potential race condition:

    if (HttpContext.Current.Cache[query] == null)         
    {   
        ...
    }         
    else         
    {
        // When you get here, another thread may have removed the item from the cache
        // so this may still return null.
        results = HttpContext.Current.Cache[query].ToString();         
    }
    

    In general I wouldn't use locking, and would do it as follows to avoid the race condition:

    results = HttpContext.Current.Cache[query];
    if (results == null)         
    {   
        results = GetResultsFromSomewhere();
        HttpContext.Current.Cache.Add(query, results,...);
    }
    return results;
    

    In the above case, multiple threads might attempt to load data if they detect a cache miss at about the same time. In practice this is likely to be rare, and in most cases unimportant, because the data they load will be equivalent.

    But if you want to use a lock to prevent it you can do so as follows:

    results = HttpContext.Current.Cache[query];
    if (results == null)         
    {   
        lock(someLock)
        {
            results = HttpContext.Current.Cache[query];
            if (results == null)
            {
                results = GetResultsFromSomewhere();
                HttpContext.Current.Cache.Add(query, results,...);
            }           
        }
    }
    return results;
    
    0 讨论(0)
提交回复
热议问题