Как очистить MemoryCache? - PullRequest
       34

Как очистить MemoryCache?

86 голосов
/ 15 ноября 2010

Я создал кеш, используя класс MemoryCache.Я добавляю в него некоторые элементы, но когда мне нужно перезагрузить кеш, сначала я хочу очистить его.Какой самый быстрый способ сделать это?Должен ли я пройтись по всем элементам и убрать их по одному или есть лучший способ?

Ответы [ 9 ]

54 голосов
/ 15 ноября 2010

Dispose существующего MemoryCache и создайте новый объект MemoryCache.

52 голосов
/ 13 марта 2014

Проблема с перечислением

Раздел примечаний MemoryCache.GetEnumerator () предупреждает: «Получение перечислителя для экземпляра MemoryCache является ресурсоемкой и блокирующей операцией. Поэтому перечислитель не должен использоваться в производственных приложениях».

Вот почему , объяснено в псевдокоде реализации GetEnumerator ():

Create a new Dictionary object (let's call it AllCache)
For Each per-processor segment in the cache (one Dictionary object per processor)
{
    Lock the segment/Dictionary (using lock construct)
    Iterate through the segment/Dictionary and add each name/value pair one-by-one
       to the AllCache Dictionary (using references to the original MemoryCacheKey
       and MemoryCacheEntry objects)
}
Create and return an enumerator on the AllCache Dictionary

Поскольку реализация разделяет кэш по нескольким объектам Dictionary, она должна собрать все в одну коллекцию, чтобы вернуть перечислитель. Каждый вызов GetEnumerator выполняет процесс полного копирования, описанный выше. Вновь созданный словарь содержит ссылки на исходные внутренние ключи и объекты-значения, поэтому фактические значения в кэше данных не дублируются.

Предупреждение в документации верно. Избегайте GetEnumerator () - включая все ответы выше, которые используют запросы LINQ.

Лучшее и более гибкое решение

Вот эффективный способ очистки кэша, который просто основан на существующей инфраструктуре мониторинга изменений. Он также обеспечивает гибкость очистки всего кэша или только именованного подмножества и не имеет проблем, описанных выше.

// By Thomas F. Abraham (http://www.tfabraham.com)
namespace CacheTest
{
    using System;
    using System.Diagnostics;
    using System.Globalization;
    using System.Runtime.Caching;

    public class SignaledChangeEventArgs : EventArgs
    {
        public string Name { get; private set; }
        public SignaledChangeEventArgs(string name = null) { this.Name = name; }
    }

    /// <summary>
    /// Cache change monitor that allows an app to fire a change notification
    /// to all associated cache items.
    /// </summary>
    public class SignaledChangeMonitor : ChangeMonitor
    {
        // Shared across all SignaledChangeMonitors in the AppDomain
        private static event EventHandler<SignaledChangeEventArgs> Signaled;

        private string _name;
        private string _uniqueId = Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture);

        public override string UniqueId
        {
            get { return _uniqueId; }
        }

        public SignaledChangeMonitor(string name = null)
        {
            _name = name;
            // Register instance with the shared event
            SignaledChangeMonitor.Signaled += OnSignalRaised;
            base.InitializationComplete();
        }

        public static void Signal(string name = null)
        {
            if (Signaled != null)
            {
                // Raise shared event to notify all subscribers
                Signaled(null, new SignaledChangeEventArgs(name));
            }
        }

        protected override void Dispose(bool disposing)
        {
            SignaledChangeMonitor.Signaled -= OnSignalRaised;
        }

        private void OnSignalRaised(object sender, SignaledChangeEventArgs e)
        {
            if (string.IsNullOrWhiteSpace(e.Name) || string.Compare(e.Name, _name, true) == 0)
            {
                Debug.WriteLine(
                    _uniqueId + " notifying cache of change.", "SignaledChangeMonitor");
                // Cache objects are obligated to remove entry upon change notification.
                base.OnChanged(null);
            }
        }
    }

    public static class CacheTester
    {
        public static void TestCache()
        {
            MemoryCache cache = MemoryCache.Default;

            // Add data to cache
            for (int idx = 0; idx < 50; idx++)
            {
                cache.Add("Key" + idx.ToString(), "Value" + idx.ToString(), GetPolicy(idx));
            }

            // Flush cached items associated with "NamedData" change monitors
            SignaledChangeMonitor.Signal("NamedData");

            // Flush all cached items
            SignaledChangeMonitor.Signal();
        }

        private static CacheItemPolicy GetPolicy(int idx)
        {
            string name = (idx % 2 == 0) ? null : "NamedData";

            CacheItemPolicy cip = new CacheItemPolicy();
            cip.AbsoluteExpiration = System.DateTimeOffset.UtcNow.AddHours(1);
            cip.ChangeMonitors.Add(new SignaledChangeMonitor(name));
            return cip;
        }
    }
}
32 голосов
/ 29 февраля 2012

С http://connect.microsoft.com/VisualStudio/feedback/details/723620/memorycache-class-needs-a-clear-method

Обходной путь:

List<string> cacheKeys = MemoryCache.Default.Select(kvp => kvp.Key).ToList();
foreach (string cacheKey in cacheKeys)
{
    MemoryCache.Default.Remove(cacheKey);
}
20 голосов
/ 07 сентября 2011
var cacheItems = cache.ToList();

foreach (KeyValuePair<String, Object> a in cacheItems)
{
    cache.Remove(a.Key);
}
9 голосов
/ 29 декабря 2013

Если производительность не является проблемой, то эта милая строчка подойдет:

cache.ToList().ForEach(a => cache.Remove(a.Key));
7 голосов
/ 11 апреля 2014

Похоже, что есть Трим метод.

Таким образом, чтобы очистить все содержимое, которое вы просто делаете

cache.Trim(100)

EDIT: после еще нескольких копаний кажется, что изучение Trim не стоит вашего времени

https://connect.microsoft.com/VisualStudio/feedback/details/831755/memorycache-trim-method-doesnt-evict-100-of-the-items

Как очистить System.Runtime.Caching.MemoryCache

3 голосов
/ 13 июля 2011

Вы также можете сделать что-то вроде этого:


Dim _Qry = (From n In CacheObject.AsParallel()
           Select n).ToList()
For Each i In _Qry
    CacheObject.Remove(i.Key)
Next
2 голосов
/ 18 июля 2014

Наткнулся на это и на его основе написал чуть более эффективный, параллельный метод очистки:

    public void ClearAll()
    {
        var allKeys = _cache.Select(o => o.Key);
        Parallel.ForEach(allKeys, key => _cache.Remove(key));
    }
0 голосов
/ 31 октября 2013

немного улучшенная версия ответа магрита.

var cacheKeys = MemoryCache.Default.Where(kvp.Value is MyType).Select(kvp => kvp.Key).ToList();
foreach (string cacheKey in cacheKeys)
{
    MemoryCache.Default.Remove(cacheKey);
}
...