• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    ASP.NET性能优化之构建自定义文件缓存
    现在,借助于.NET4.0中的OutputCacheProvider,我们可以有多种选择创建自己的缓存。如,我们可以把HTML输出缓存存储到memcached分布式集群服务器,或者MongoDB中(一种常用的面向文档数据库,不妨阅读本篇http://msdn.microsoft.com/zh-cn/magazine/gg650661.aspx)。当然,我们也可以把缓存作为文件存储到硬盘上,考虑到可扩展性,这是一种最廉价的做法,本文就是介绍如果构建自定义文件缓存。

    1:OutputCacheProvider

    OutputCacheProvider是一个抽象基类,我们需要override其中的四个方法,它们分别是:

    Add 方法,将指定项插入输出缓存中。

    Get 方法,返回对输出缓存中指定项的引用。

    Remove 方法,从输出缓存中移除指定项。

    Set 方法,将指定项插入输出缓存中,如果该项已缓存,则覆盖该项。

    2:创建自己的文件缓存处理类

    该类型为FileCacheProvider,代码如下:

    复制代码 代码如下:

    public class FileCacheProvider : OutputCacheProvider
    {
    private static readonly ILog log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
    public override void Initialize(string name, NameValueCollection attributes)
    {
    base.Initialize(name, attributes);
    CachePath = HttpContext.Current.Server.MapPath(attributes["cachePath"]);
    }
    public override object Add(string key, object entry, DateTime utcExpiry)
    {
    Object obj = Get(key);
    if (obj != null) //这一步很重要
    {
    return obj;
    }
    Set(key,entry,utcExpiry);
    return entry;
    }
    public override object Get(string key)
    {
    string path = ConvertKeyToPath(key);
    if (!File.Exists(path))
    {
    return null;
    }
    CacheItem item = null;
    using (FileStream file = File.OpenRead(path))
    {
    var formatter = new BinaryFormatter();
    item = (CacheItem)formatter.Deserialize(file);
    }
    if (item.ExpiryDate = DateTime.Now.ToUniversalTime())
    {
    log.Info(item.ExpiryDate + "*" + key);
    Remove(key);
    return null;
    }
    return item.Item;
    }
    public override void Set(string key, object entry, DateTime utcExpiry)
    {
    CacheItem item = new CacheItem(entry, utcExpiry);
    string path = ConvertKeyToPath(key);
    using (FileStream file = File.OpenWrite(path))
    {
    BinaryFormatter formatter = new BinaryFormatter();
    formatter.Serialize(file, item);
    }
    }
    public override void Remove(string key)
    {
    string path = ConvertKeyToPath(key);
    if (File.Exists(path))
    File.Delete(path);
    }
    public string CachePath
    {
    get;
    set;
    }
    private string ConvertKeyToPath(string key)
    {
    string file = key.Replace('/', '-');
    file += ".txt";
    return Path.Combine(CachePath, file);
    }
    }
    [Serializable]
    public class CacheItem
    {
    public DateTime ExpiryDate;
    public object Item;
    public CacheItem(object entry, DateTime utcExpiry)
    {
    Item = entry;
    ExpiryDate = utcExpiry;
    }
    }

    有两个地方需要特别说明:
    在Add方法中,有一个条件判断,必须做出这样的处理,否则缓存机制将会缓存第一次的结果,过了有效期后缓存讲失效并不再重建;
    在示例程序中,我们简单的将缓存放到了Cache目录下,在实际的项目实践中,考虑到缓存的页面将是成千上万的,所以我们必须要做目录分级,否则寻找并读取缓存文件将会成为效率瓶颈,这会耗尽CPU。
    3:配置文件
    我们需要在Web.config中配置缓存处理程序是自定义的FileCacheProvider,即在 system.web>下添加节点:
    复制代码 代码如下:

    caching>
    outputCache defaultProvider="FileCache">
    providers>
    add name="FileCache" type="MvcApplication2.Common.FileCacheProvider" cachePath="~/Cache" />
    /providers>
    /outputCache>
    /caching>

    4:缓存的使用
    我们假设在MVC的控制中使用(如果要在ASP.NET页面中使用,则在页面中包含%@OutputCache VaryByParam="none" Duration="10" %>),可以看到,Index是未进行输出缓存的,而Index2进行了输出缓存,缓存时间为10秒。
    复制代码 代码如下:

    public class HomeController : Controller
    {
    private static readonly ILog log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
    static string s_conn = "Data Source=192.168.0.77;Initial Catalog=luminjidb;User Id=sa;Password=sa;";
    public ActionResult Index()
    {
    using (DataSet ds = Common.SqlHelper.ExecuteDataset(s_conn, CommandType.Text, "select top 1* from NameTb a, DepTb b where a.DepID = b.ID ORDER BY NEWID()"))
    {
    ViewBag.Message = ds.Tables[0].Rows[0]["name"].ToString();
    }
    return View();
    }
    [OutputCache(Duration = 10, VaryByParam = "none")]
    public ActionResult Index2()
    {
    using (DataSet ds = Common.SqlHelper.ExecuteDataset(s_conn, CommandType.Text, "select top 1* from NameTb a, DepTb b where a.DepID = b.ID ORDER BY NEWID()"))
    {
    ViewBag.Message = ds.Tables[0].Rows[0]["name"].ToString();
    }
    return View();
    }
    }

    5:查看下效果

    上面的代码,在访问了Index2后,将会在Cache文件夹下产生缓存文件,如下:

    image

    现在,我们开始评价下有输出缓存和无输出缓存的性能对比,模拟100个用户并发1000次请求如下:

    image

    可以看到,有输出缓存后,吞吐率明显提高了10倍。

    6:代码下载

    FileCacheProvider的原始代码来自于网络,我修改了其中的BUG,全部代码下载如下:MvcApplication20110907.rar

    您可能感兴趣的文章:
    • ASP.NET性能优化之局部缓存分析
    • ASP.NET 性能优化之反向代理缓存使用介绍
    • ASP.NET性能优化之让浏览器缓存动态网页的方法
    • ASP.NET性能优化小结(ASP.NETC#)
    • asp.net 程序性能优化的七个方面 (c#(或vb.net)程序改进)
    • ASP.NET比较常用的26个性能优化技巧
    • asp.net小谈网站性能优化
    • ASP.NET性能优化之减少请求
    • ASP.NET技巧:同时对多个文件进行大量写操作对性能优化
    • asp.net性能优化之使用Redis缓存(入门)
    上一篇:ASP.NET WebService中使用ASP.NET_SessionId的问题说明
    下一篇:ASP.NET性能优化之让浏览器缓存动态网页的方法
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    ASP.NET性能优化之构建自定义文件缓存 ASP.NET,性能,优化,之,构建,