c# - 这是清除 C# MemoryCache 的好解决方案吗?

标签 c# multithreading caching memorycache

我已阅读我能找到的有关在 C# 中清除 MemoryCache 的问题和答案。 有很多建议,例如: 1.枚举缓存,并删除所有项目-根据其他人的说法,这不好,获取枚举器会锁定整个事物,并且会发生各种天启,引用文档的一部分,我还没有找到,并且显示警告,我未能重现。无论如何,我认为这不是一个非常有效的解决方案。

  1. 将键存储在一个单独的集合中,并迭代该集合,以从缓存中删除项目。 - 除了这听起来不是很线程安全,它听起来也不是很有效。

  2. 处理旧缓存,并创建一个新缓存 - 这听起来不错,这是我想到的明显问题,并在几条评论中指出,对旧缓存的现有引用可能会带来问题。当然,操作的顺序很重要,您必须为旧的保存一个引用,创建一个新的来代替旧的,然后处置旧的 - 似乎并不是每个人都注意到这个细微差别。

那现在呢?我的一位同事建议使用 MemoryCache 来解决我的问题,他将缓存对象包装在另一个类中,该类可以选择获取 key (如果需要,从 db 加载它)、删除 key 或清除缓存.前两个现在不重要,第三个很有趣。我为此使用了第三种解决方案,按照“保证,除了我自己的实现之外,没有对 MemoryCache 对象的任何其他引用”。所以相关代码是:

构造函数:

public MyCache()
{
    _cache = new MemoryCache("MyCache");
}

清除缓存:

public void ClearCacheForAllUsers()
{
    var oldCache = _cache;
    _cache = new MemoryCache("MyCache");
    oldCache.Dispose();
}

_cache 是私有(private) MemoryCache 对象。

这会在多线程环境中引起问题吗?我担心并行调用读取和处理。我是否应该实现某种锁定机制,允许并发读取,但会导致清除缓存功能等待当前读取缓存完成?

我的猜测是肯定的,它是实现它所必需的,但我想在开始之前先了解一下。

罗伯特

对 Voo 的回答进行编辑 1: 可以保证,“包装器”(MyCache)之外的任何人都不会获得对 _cache 对象的引用。我担心的是以下内容:

T1:
MyCache.ClearAll()
    var oldCache = _cache
T2:
MyCache.GetOrReadFromDb("stuff")
T1:
    _cache=new MemoryCache("MyCache")
    oldCache.Dispose()
T2:
    *something bad*

除了仍然使用旧缓存的 T2 线程(这不是首选,但我可以接受)之外,是否存在 Get 方法以某种方式访问​​处于处置状态的旧缓存或读取新缓存的情况没有数据?

想象一下 GetOrReadFromDb 函数是这样的:

public object GetOrReadFromDb(string key)
{
    if(_cache[key]==null)
    {
        //Read stuff from DB, and insert into the cache
    }
    return _cache[key];
}

我认为有可能,控制从读取线程中拿走,并交给清理,例如在从数据库读取之后,在返回 _chache[key] 值之前,这可能会导致问题.这是一个真正的问题吗?

最佳答案

您的解决方案的问题是它引入了可能的竞争条件,需要难看的锁或其他线程同步解决方案。

改为使用内置解决方案。

您可以使用自定义 ChangeMonitor 类从缓存中清除您的项目。您可以将 ChangeMonitor 实例添加到您在调用 MemoryCache.Set 时设置的 CacheItemPolicyChangeMonitors 属性。

例如:

class MyCm : ChangeMonitor
{
    string uniqueId = Guid.NewGuid().ToString();

    public MyCm()
    {
        InitializationComplete();
    }

    protected override void Dispose(bool disposing) { }

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

    public void Stuff()
    {
        this.OnChanged(null);
    }
}

使用示例:

        var cache = new MemoryCache("MyFancyCache");
        var obj = new object();
        var cm = new MyCm();
        var cip = new CacheItemPolicy();
        cip.ChangeMonitors.Add(cm);

        cache.Set("MyFancyObj", obj, cip);

        var o = cache.Get("MyFancyObj");
        Console.WriteLine(o != null);

        o = cache.Get("MyFancyObj");
        Console.WriteLine(o != null);

        cm.Stuff();

        o = cache.Get("MyFancyObj");
        Console.WriteLine(o != null);

        o = cache.Get("MyFancyObj");
        Console.WriteLine(o != null);

关于c# - 这是清除 C# MemoryCache 的好解决方案吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25269338/

相关文章:

c# - Xamarin.iOS 应用程序在事件时不接收 parse.com PUSH

c# - 我可以在 CentOS 上安装适用于 Linux 的 .NET Core 和 Npgsql 吗?

c# - 类型 'System.Int32' 的表达式不能用于方法 'System.Object' 的类型 'Boolean Equals(System.Object)' 的参数

linux - Linux 和 Windows 中的多线程模型

c# - 无法在 .zip 中导出模板 visual Studio 2010

java - 静态同步方法

c++ - 弹出对话框将文件保存在单独的线程中

android - 缓存位图 - 检查当前 VM 预算

android - RxJava 和缓存数据

iOS Mobile Web App(通过将页面保存到主屏幕)离线可用性