欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文搞懂MemoryCache 清除全部緩存的方法

 更新時間:2021年12月22日 09:42:15   作者:菜鳥耕地  
這篇文章主要介紹了MemoryCache 清除全部緩存的方法,本文給大家介紹兩個方法,每種方法通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

最近有個需求需要定時清理服務(wù)器上所有的緩存。本來以為很簡單的調(diào)用一下 MemoryCache.Clear 方法就完事了。誰知道 MemoryCache 類以及 IMemoryCache 擴展方法都沒有 Clear 方法。這可給難住了,于是想找到所有的 Keys 來一個個 Remove ,誰知道居然也沒有獲取所有 Key 的方法。于是研究了一下 ,找到一些方法,下面介紹兩個方法:

自定義 CacheWrapper 包裝類

MemoryCache 構(gòu)造 Entry 的時候支持傳入 CancellationChangeToken 對象,當(dāng) CancellationChangeToken.Cancel 觸發(fā)的時候會自動使該對象過期。那么我們只要對 MemoryCache 類包裝一下很容易實現(xiàn)一個自己的 Cache 類。

 public class CacheWrapper
    {
        private readonly IMemoryCache _memoryCache;
        private CancellationTokenSource _resetCacheToken = new();

        public CacheWrapper(IMemoryCache memoryCache)
        {
            _memoryCache = memoryCache;
        }

        public void Add(object key, object value, MemoryCacheEntryOptions memoryCacheEntryOptions)
        {
            using var entry = _memoryCache.CreateEntry(key);
            entry.SetOptions(memoryCacheEntryOptions);
            entry.Value = value;

            // add an expiration token that allows us to clear the entire cache with a single method call
            entry.AddExpirationToken(new CancellationChangeToken(_resetCacheToken.Token));
        }

        public object Get(object key)
        {
            return _memoryCache.Get(key);
        }

        public void Remove(object key)
        {
            _memoryCache.Remove(key);
        }

        public void Clear()
        {
            _resetCacheToken.Cancel(); // this triggers the CancellationChangeToken to expire every item from cache

            _resetCacheToken.Dispose(); // dispose the current cancellation token source and create a new one
            _resetCacheToken = new CancellationTokenSource();
        }
    }

然后單元測試測試一下:

        [TestMethod()]
        public void ClearTest()
        {
            var memCache = new MemoryCache(new MemoryCacheOptions());
            var wrapper = new CacheWrapper(memCache);

            for (int i = 0; i < 10; i++)
            {
                wrapper.Add(i.ToString(), new object(), new MemoryCacheEntryOptions());
            }

            Assert.IsNotNull(wrapper.Get("1"));
            Assert.IsNotNull(wrapper.Get("9"));

            wrapper.Clear();

            for (int i = 0; i < 10; i++)
            {
                Assert.IsNull(wrapper.Get(i.ToString()));
            }

            for (int i = 0; i < 10; i++)
            {
                wrapper.Add(i.ToString(), new object(), new MemoryCacheEntryOptions());
            }

            Assert.IsNotNull(wrapper.Get("1"));
            Assert.IsNotNull(wrapper.Get("9"));

            wrapper.Clear();

            for (int i = 0; i < 10; i++)
            {
                Assert.IsNull(wrapper.Get(i.ToString()));
            }

        }

測試通過。

Compact 方法

以上 CacheWrapper 類雖然可以實現(xiàn)我們想要的功能,但是對于原來的程序有侵入,需要使用 CacheWrapper 類替換默認(rèn)的 MemoryCache 類,不是太好。于是不死心繼續(xù)研究,后來直接看了 MemoryCache 的代碼(源碼在這),開源真香。發(fā)現(xiàn) MemoryCache 有個 Compact 方法好像在干刪除的勾當(dāng)。也怪我英文不好,這單詞是壓縮的意思,居然才發(fā)現(xiàn)。。。。于是我們的清除所有對象的需求不就輕而易舉了么?

 /// Remove at least the given percentage (0.10 for 10%) of the total entries (or estimated memory?), according to the following policy:
        /// 1. Remove all expired items.
        /// 2. Bucket by CacheItemPriority.
        /// 3. Least recently used objects.
        /// ?. Items with the soonest absolute expiration.
        /// ?. Items with the soonest sliding expiration.
        /// ?. Larger objects - estimated by object graph size, inaccurate.
MemoryCache.Compact(double percentage);

Compact 方法會對緩存的對象進行壓縮,參數(shù)是個double,0.1 表示壓縮 10% ,那么傳 1.0 就是壓縮 100%,那不就是 Clear All 么。所以我可以使用 Compact(1.0) 來清除所有的緩存對象。單元測試跑一下:

[TestMethod()]
        public void CompactTest()
        {
            var memCache = new MemoryCache(new MemoryCacheOptions());

            for (int i = 0; i < 10; i++)
            {
                memCache.Set(i.ToString(), new object(), new MemoryCacheEntryOptions());
            }

            Assert.IsNotNull(memCache.Get("1"));
            Assert.IsNotNull(memCache.Get("9"));

            memCache.Compact(1);

            for (int i = 0; i < 10; i++)
            {
                Assert.IsNull(memCache.Get(i.ToString()));
            }

            for (int i = 0; i < 10; i++)
            {
                memCache.Set(i.ToString(), new object(), new MemoryCacheEntryOptions());
            }

            Assert.IsNotNull(memCache.Get("1"));
            Assert.IsNotNull(memCache.Get("9"));

            memCache.Compact(1);

            for (int i = 0; i < 10; i++)
            {
                Assert.IsNull(memCache.Get(i.ToString()));
            }
        }

完美通過。
這里簡單介紹下 Compact 方法。根據(jù)注釋它會按照已下優(yōu)先級刪除對象:

  • 過期的對象
  • CacheItemPriority 設(shè)置的優(yōu)先級,等級越高越不容易被刪除
  • 最近最少被使用的對象
  • 絕對過期時間
  • 滑動過期時間
  • 大對象

到此這篇關(guān)于一文搞懂MemoryCache 清除全部緩存的文章就介紹到這了,更多相關(guān)MemoryCache 清除全部緩存內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論