ASP.NET MVC 和 MemoryCache - 我如何使用它?

Posted

技术标签:

【中文标题】ASP.NET MVC 和 MemoryCache - 我如何使用它?【英文标题】:ASP.NET MVC and MemoryCache - how do i use it? 【发布时间】:2013-01-31 16:22:18 【问题描述】:

我的 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 总是为空,我做错了什么?

【问题讨论】:

查看deanhume.com/home/blogpost/object-caching----net-4/37 【参考方案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 等于您要使用的实例。

以上是关于ASP.NET MVC 和 MemoryCache - 我如何使用它?的主要内容,如果未能解决你的问题,请参考以下文章

使用内存缓存时 Asp.Net MVC 中的 InvalidOperationException

在 asp.net 核心的嵌套函数中使用 MemoryCache

如何在 asp.net core 中遍历 MemoryCache?

缓存处理类(MemoryCache结合文件缓存)

Asp.net mvc和asp.net有啥区别?

Asp.net mvc和asp.net有啥区别?