【问题标题】:ASP.NET MVC and MemoryCache - how do i use it?ASP.NET MVC 和 MemoryCache - 我如何使用它?
【发布时间】:2026-01-09 01:10:01
【问题描述】:

我的 Application_Start 中有这个:

var crumbsCache = new MemoryCache("breadCrumbsNames");
var crumbsList = new List<CacheItem>
                    {
                        //list of new CacheItem();
                    };
foreach (var cacheItem in crumbsList)
{
    crumbsCache.Add(cacheItem, new CacheItemPolicy());
}

现在,在我的控制器中,我正在这样做:

var cache = new MemoryCache("breadCrumbsNames");
var cacheItem = cache.GetCacheItem("nameOfCacheItem");

但是 cacheItem 总是为空,我做错了什么?

【问题讨论】:

标签: c# asp.net asp.net-mvc memorycache


【解决方案1】:

我认为对您来说更好的选择是使用Ninject 或其他一些依赖注入框架根据需要将您的MemoryCache 注入控制器。

您将首先将 NinjectNinject.Mvc3(以及任何其他相关位)添加到您的 ASP.NET MVC 项目中。如果您在 Visual Studio 中工作,则可以使用 NuGet 来执行此操作。它非常轻松且自动化程度很高。

下一步是将您的MemoryCache 包装到某种接口中,例如:

public interface IMemoryCacheService
{
    MemoryCache MemoryCache
    {
        get;
        set;
    }
}

还有:

public class MemoryCacheService : IMemoryCacheService
{
    public MemoryCacheService()
    {
        MemoryCache = new MemoryCache();
    }

    public MemoryCache MemoryCache
    {
        get;
        set;
    }
}

然后您在 Ninject 中定义一个绑定,以便 Ninject 知道当您需要 IMemoryCacheService 类型的东西时,它应该为您提供 MemoryCacheService 的实例。

我将在此处粘贴我自己的 Ninject 配置类。将在您的项目中创建的将非常相似,并将位于名为App_Start 的文件夹中(如果您使用 NuGet,它将自动创建)。 Ninject 默认创建的类名为NinjectWebCommon

public static class NinjectConfig
{
    private static readonly Bootstrapper bootstrapper = new Bootstrapper();

    public static void Start()
    {
        DynamicModuleUtility.RegisterModule(typeof(OnePerRequestHttpModule));
        DynamicModuleUtility.RegisterModule(typeof(NinjectHttpModule));

        bootstrapper.Initialize(CreateKernel);
    }

    public static void Stop()
    {
        bootstrapper.ShutDown();
    }

    private static IKernel CreateKernel()
    {
        var kernel = new StandardKernel();

        kernel.Bind<Func<IKernel>>()
              .ToMethod(context => () => new Bootstrapper().Kernel);
        kernel.Bind<IHttpModule>()
              .To<HttpApplicationInitializationHttpModule>();
        kernel.RegisterServices();

        return kernel;
    }

    private static void RegisterServices(this IKernel kernel)
    {
        kernel.Bind<IMemoryCacheService>()
              .To<MemoryCacheService>()
              .InSingletonScope();
              // InSingletonScope() is important so Ninject knows
              // to create only one copy and then reuse it every time
              // it is asked for

        // ignore the stuff below... I have left it in here for illustration
        kernel.Bind<IDbTransactionFactory>()
              .To<DbTransactionFactory>()
              .InRequestScope();
        kernel.Bind<IDbModelContext>()
              .To<DbModelContext>()
              .InRequestScope();
        kernel.Bind<IDbModelChangeContext>()
              .To<DbModelChangeContext>()
              .InRequestScope();
        kernel.Bind<IUserContext>()
              .To<UserContext>()
              .InRequestScope();

        kernel.BindAttributeAndFilter<IgnoreNonAjaxRequestsFilter, IgnoreNonAjaxRequestsAttribute>();
        kernel.BindAttributeAndFilter<ProvideApplicationInfoFilter, ProvideApplicationInfoAttribute>();
        kernel.BindAttributeAndFilter<ProvideSessionInfoFilter, ProvideSessionInfoAttribute>();
        kernel.BindAttributeAndFilter<UseDialogLayoutFilter, UseDialogLayoutAttribute>();
        kernel.BindAttributeAndFilter<CheckResourceAccessFilter, CheckResourceAccessAttribute>();
        kernel.BindAttributeAndFilter<CheckResourceStateFilter, CheckResourceStateAttribute>();
    }

    private static void BindAttributeAndFilter<TFilter, TAttribute>(this IKernel kernel)
    {
        kernel.BindFilter<TFilter>(FilterScope.Action, null)
              .WhenControllerHas<TAttribute>();
        kernel.BindFilter<TFilter>(FilterScope.Action, null)
              .WhenActionMethodHas<TAttribute>();
    }
}

最后,您的控制器将从:

public class HomeController : Controller
{
    public ActionResult Foo()
    {
        ...
    }

    ...
}

到:

public class HomeController : Controller
{
    private IMemoryCacheService memoryCacheService;

    public HomeController(IMemoryCacheService memoryCacheService)
    {
        this.memoryCacheService = memoryCacheService;
    }

    public ActionResult Foo()
    {
        // use this.memoryCacheService in your controller methods...
    }

    ...
}

假设您按照上述策略创建了另一个名为IEmailService 的服务,并且您希望IEmailService 也可以在HomeController 中使用,那么:

public class HomeController : Controller
{
    private IMemoryCacheService memoryCacheService;
    private IEmailService emailService;

    public HomeController(IMemoryCacheService memoryCacheService, IEmailService emailService)
    {
        this.memoryCacheService = memoryCacheService;
        this.emailService = emailService;
    }

    public ActionResult Foo()
    {
        // use this.memoryCacheService in your controller methods...
        // and also use this.emailService in your controller methods...
    }

    ...
}

Ninject 将更改 ASP.NET MVC 控制器工厂以自动将注入的参数提供给控制器构造函数。

我认为从长远来看,这种方法比保留全局变量等更好。

【讨论】:

  • 我认为在大多数情况下,这增加了比价值更多的开销。 [编辑添加:这可能适用于分布式缓存。]只需使用 Ryan Byrne 描述的 MemoryCache.Default。
  • 老实说,我写以上所有内容更多是为了说明正确的方法,而不是回答具体问题。核心问题是 OP 试图在控制器中创建一个对象(这将是每个请求),而他需要一个单例。一般来说,我们希望注入依赖项,而不是使用静态类/属性。
  • 此外,如果您阅读了 OP 对 Ryan Byrne 的回答的评论,很明显(OP 可能不知道)OP 正在寻找 DI 机制,而不仅仅是正确初始化全局变量。
  • 感谢 Umar 的彻底回答!
【解决方案2】:

您正在每个控制器中创建MemoryCache 的新实例。由于它是新的,因此其中没有任何内容,这就是为什么您的值始终为空的原因。您需要访问在Application_Start 中创建的同一实例。考虑使用MemoryCache.Default

【讨论】:

  • 但是MemoryCache.Default到底是什么?如果我要设置两个不同的 memcache 实例,MemoryCache.Default 怎么知道哪个是默认的?
  • 设置 MemoryCache.Default 等于您要使用的实例。
最近更新 更多