在类库中正确实现缓存以在asp.net应用程序中使用

时间:2008-10-01 06:38:26

标签: c# asp.net caching garbage-collection singleton

我正在我在asp.net应用程序中使用的类库中实现缓存。

我使用静态方法创建了我的缓存对象作为单例模式来更新缓存,这实际上只是加载一个成员变量/属性,其中包含我需要缓存的数据集合(得到一些锁定逻辑的课程)。我认为这是一个很好的方式,因为我可以通过调用

来访问我的数据
MyCacheObject.Instance.MyDataCollection

我正在创建一个新的缓存对象来存储由某个键分区的大量数据。我所说的是我正在创建一个新的缓存,但是这个不会一次加载所有数据,而是为每个访问的密钥存储一个集合。

MyOtherCacheObject.Instance.MyOtherDataCollection(indexkey)

这次提出了关于垃圾收集的问题。由于我存储了大量的数据,如果突然得到gc,它不会是浪费吗?由于它只是一个单例模式,因此无法确保数据保留在缓存中。

所以我的问题是 - 实现缓存来处理这种情况的最佳做法是什么?我真的不喜欢这个非常复杂的解决方案,我知道System.Web中有缓存,但这似乎有点“关闭”,因为这只是一个类库,或者你怎么看?

2 个答案:

答案 0 :(得分:9)

在我看来,最佳解决方案具有以下特征:

  • 使用平台提供的可用缓存服务,试图避免编写自己的缓存服务。

  • 不将您的类库与System.Web结合,以使图层保持一致。

  • 但是如果类库在ASP.NET应用程序中运行,则解决方案不应该要求另外的缓存实现(例如,企业库缓存应用程序块),这需要额外的配置和设置。 / p>

因此,我将使用IoC策略,以允许类库根据其运行的环境使用不同的缓存实现。

假设您将抽象缓存合约定义为:

public interface ICacheService 
{
    AddItem(...);
}

您可以提供基于System.Web的实现:

public AspNetBasedCacheService : ICacheService
{
    AddItem(...)
    {
        // Implementation that uses the HttpContext.Cache object
    }
 }

然后将该实现“发布”为单例。请注意,与原始方法的不同之处在于单例只是对基于ASP.NET缓存服务的实现的引用,而不是完整的“缓存对象”。

public class ChacheServiceProvider 
{
    public static IChacheService Instance {get; set;}

}

您必须通过执行延迟初始化或在应用程序启动时(在global.asax.cs中)初始化chaching实现

每个域组件都可以使用已发布的缓存服务,而无需知道它是基于System.Web实现的。

// inside your class library:
IChacheService chache = CacheServiceProvider.Instance;
cache.AddItem(...);

我同意它可能不是最简单的解决方案,但我的目标是利用ASP.NET缓存实现而不会牺牲代码解耦和灵活性。

我希望我理解你的问题。

答案 1 :(得分:1)

只要缓存仍然拥有对它的引用,数据就不会被垃圾收集。

此外,不要使用单身人士。