1:OutputCacheProvider web
OutputCacheProvider是一個抽象基類,咱們須要override其中的四個方法,它們分別是: 數據庫
Add 方法,將指定項插入輸出緩存中。 緩存
Get 方法,返回對輸出緩存中指定項的引用。 服務器
Remove 方法,從輸出緩存中移除指定項。 網絡
Set 方法,將指定項插入輸出緩存中,若是該項已緩存,則覆蓋該項。 併發
2:建立本身的文件緩存處理類 分佈式
該類型爲FileCacheProvider,代碼以下: ide
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;
}
}
|
有兩個地方須要特別說明: memcached
在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