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文件夾下產生緩存文件,如下:
現在,我們開始評價下有輸出緩存和無輸出緩存的性能對比,模擬100個用戶並發1000次請求如下:
可以看到,有輸出緩存後,吞吐率明顯提高了10倍。
6:代碼下載
FileCacheProvider的原始代碼來自於網絡,我修改了其中的BUG,全部代碼下載如下:MvcApplication20110907.rar