ASP.NET výstupné cache (t. j. statické HTML) bolo založené na pamäti až do .NET 4.0. To znamená, že ak naša stránka obsahuje veľa cache, je ľahké spotrebovať lokálnu pamäť. Teraz, s pomocou . OutputCacheProvider v .NET 4.0 máme niekoľko možností, ako vytvoriť vlastnú cache. Napríklad môžeme uložiť výstupnú HTML cache v memcached distribuovanom klastrovom serveri alebo MongoDB (bežne používaná dokumentovo orientovaná databáza, čítaj tento http://msdn.microsoft.com/zh-cn/magazine/gg650661.aspx). Samozrejme, cache môžeme tiež uložiť ako súbor na pevný disk, čo je najlacnejší spôsob vzhľadom na škálovateľnosť, a tento článok je o tom, ako vytvoriť vlastnú cache súborov.
1:OutputCacheProvider OutputCacheProvider je abstraktná základná trieda, ktorú musíme prepísať štyri z jej metód, a to sú: Pridajte metódu na vloženie špecifikovanej položky do výstupnej cache. Získajte metódu, ktorá vracia odkaz na zadaný predmet vo výstupnej cache. Odstrániť metódu na odstránenie špecifikovanej položky z výstupnej cache. Metóda Set vloží špecifikovanú položku do výstupnej cache a prepíše ju, ak je uložená v cache.
2: Vytvorte si vlastnú triedu na spracovanie cache súborov Typ je FileCacheProvider a kód je nasledovný:
- 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;
- }
- }
Kopírovať kód Existujú dve miesta, ktoré si vyžadujú osobitnú pozornosť: Pri metóde Add je potrebné vykonať podmienené hodnotenie týmto spôsobom, inak mechanizmus cache uloží prvý výsledok do vyrovnávacej pamäte a cache vyprší po uplynutí platnosti a nebude znovu vytvorená. V príkladovom programe jednoducho umiestnime cache do cache adresára a v reálnej projektovej praxi, keďže cache stránok budú tisíce, musíme robiť klasifikáciu adresárov, inak by sa vyhľadávanie a čítanie cache súborov stalo úzkym hrdlom pre efektivitu, čo by vyčerpalo CPU.
3: Konfiguračný súbor
Musíme v Web.config nakonfigurovať, že cache handler je vlastný FileCacheProvider, teda pridať uzol pod FileCacheProvider:- <caching>
- <outputCache defaultProvider="FileCache">
- <providers>
- <add name="FileCache" type="MvcApplication2.Common.FileCacheProvider" cachePath="~/Cache" />
- </providers>
- </outputCache>
- </caching>
Kopírovať kód
4: Použitie cache
Predpokladáme, že pri použití v kontrole MVC (ak ho chcete použiť na ASP.NET stránke, zahrňte < % do page@OutputCache VaryByParam="žiadne" Duration="10" %>), a vidíte, že Index nie je uložený v cache, zatiaľ čo Index2 je 10 sekúnd v cache.
- 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();
- }
- }
Kopírovať kód 5: Skontrolujte efekt Vyššie uvedený kód, po prístupe k Index2, vygeneruje cache súbor v priečinku Cache, nasledovne:
Teraz zhodnoťme porovnanie výkonu medzi výstupnou a výstupnou cache, simulujúc 100 súbežných požiadaviek od 100 používateľov nasledovne:
|