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

本文介绍如何在ASP.NET中利用.NET 4.0的OutputCacheProvider特性创建自定义文件缓存,通过实例演示如何实现缓存的增删改查,并展示其在提高网站性能方面的显著效果。

ASP.NET的输出缓存(即静态HTML)在.NET4.0前一直是基于内存的。这意味着如果我们的站点含有大量的缓存,则很容易消耗掉本机内存。现在,借助于.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

内容概要:本文系统阐述了智能物流路径规划的技术体系与实践应用,涵盖其发展背景、核心问题建模、关键算法、多目标与动态环境处理、系统架构及典型应用场景。文章以车辆路径问题(VRP)及其变体为核心数学模型,介绍了从Dijkstra、A*等单智能体算法到多车VRP的元启发式求解方法(如遗传算法、蚁群算法、大规模邻域搜索),并深入探讨了多目标优化(成本、时间、碳排放)与动态环境(实时订单、交通变化)下的自适应规划策略。结合城市配送、干线运输、场内物流等案例,展示了路径规划在提升效率、降低成本方面的实际价值,并分析了当前面临的复杂性、不确定性等挑战,展望了AI融合、数字孪生、车路协同等未来趋势。; 适合人群:具备一定物流、运筹学或计算机基础,从事智能交通、物流调度、算法研发等相关工作的技术人员与管理人员,工作年限1-5年为宜。; 使用场景及目标:①理解智能物流路径规划的整体技术架构与核心算法原理;②掌握VRP建模方法与多目标、动态环境下路径优化的实现策略;③为物流系统设计、算法选型与系统优化提供理论依据与实践参考; 阅读建议:建议结合文中案例与数学模型,重点理解算法选择与实际业务场景的匹配逻辑,关注动态规划与多目标优化的工程实现难点,可配合仿真工具或开源求解器进行实践验证。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值