.NET 緩存模塊設(shè)計(jì)實(shí)踐
上一篇談了我對(duì)緩存的概念,框架上的理解和看法,這篇承接上篇講講我自己的緩存模塊設(shè)計(jì)實(shí)踐。
基本的緩存模塊設(shè)計(jì)
最基礎(chǔ)的緩存模塊一定有一個(gè)統(tǒng)一的CacheHelper,如下:
public interface ICacheHelper
{
T Get<T>(string key);
void Set<T>(string key, T value);
void Remove(string key);
}
然后業(yè)務(wù)層是這樣調(diào)用的
public User Get(int id)
{
if (id <= 0)
throw new ArgumentNullException("id");
var key = string.Format(USER_CACHE_KEY, id);
var user = _cacheHelper.Get<User>(key);
if (user != null)
return user;
return _repository.Get(id);
}
上面的代碼沒(méi)什么錯(cuò)誤,但是實(shí)際運(yùn)用的時(shí)候就產(chǎn)生疑問(wèn)了,因?yàn)槲乙恢睆?qiáng)調(diào)緩存要保存"熱數(shù)據(jù)",那樣"熱數(shù)據(jù)"一定會(huì)有過(guò)期的時(shí)候,我們不可能另外寫一個(gè)去Set。所以干脆就結(jié)合到一起寫是比較合適的。
public User GetV2(int id)
{
if (id <= 0)
throw new ArgumentNullException("id");
var key = string.Format(USER_CACHE_KEY, id);
var user = _cacheHelper.Get<User>(key);
if (user != null)
return user;
user = _repository.Get(id);
if (user != null)
_cacheHelper.Set(key, user);
return user;
}
上面的代碼其實(shí)只是加了一個(gè)Set而已,就這樣的設(shè)計(jì)的話,每次一個(gè)Get需要的重復(fù)代碼實(shí)在是太多了,那么是不是應(yīng)該更精簡(jiǎn)?這時(shí)候吃點(diǎn)C#語(yǔ)法糖就很有必要了,語(yǔ)法糖偶爾吃點(diǎn)增進(jìn)效率,何樂(lè)而不為?
public User GetV3(int id)
{
if (id <= 0)
throw new ArgumentNullException("id");
var key = string.Format(USER_CACHE_KEY, id);
return _cacheHelperV2.Get<User>(key, () => _repository.Get(id));
}
//ICache Get<T>實(shí)現(xiàn)
public T Get<T>(string key, Func<T> fetch = null)
{
T result = default(T);
var obj = Cache.Get(key);
if (obj is T)
{
result = (T)obj;
}
if(result == null)
{
result = fetch();
if (result != null)
Set(key, result);
}
return result;
}
這里我直接把Set方法都包裝進(jìn)了ICache.Get<T>,附帶上Fetch Func。這樣就把公共的操作抽象到了一起,簡(jiǎn)化了Cache的調(diào)用,完美的符合了我的想法。
緩存模塊設(shè)計(jì)進(jìn)階
上一節(jié)里的ICache V3幾乎已經(jīng)最精簡(jiǎn)了,但是其實(shí)參考了ServiceStack.Redis之后,我發(fā)現(xiàn)了更加的抽象方式。很明顯上一節(jié)的所有代碼里,都是手動(dòng)管理Key的,對(duì)于通常的對(duì)象Cache,這個(gè)Key還需要手動(dòng)嗎?來(lái)上最后一份改進(jìn)。
public T Get<T>(object id, Func<T> fetch = null)
{
var type = typeof(T);
var key = string.Format("urn:{1}:{2}", type.Name, id.ToString());//這里是關(guān)鍵,直接用TypeName來(lái)充當(dāng)Key
return Get(key, fetch);
}
public T Get<T>(string key, Func<T> fetch = null)
{
T result = default(T);
var obj = Cache.Get(key);
if (obj is T)
{
result = (T)obj;
}
if (result == null)
{
result = fetch();
if (result != null)
Set(key, result);
}
return result;
}
Get方法完全自動(dòng)化管理了Key,然后調(diào)用的方式再次被精簡(jiǎn)。
public User GetV4(int id)
{
if (id <= 0)
throw new ArgumentNullException("id");
return _cacheHelperV3.Get<User>(id, () => _repository.Get(id));
}
很明顯還少了最重要的Set啊,Set的時(shí)候這個(gè)Key獲取就要費(fèi)一點(diǎn)事情了,最需要 解決的是如何獲取這個(gè)主鍵id的值。
public class User
{
[PrimaryKey] //這個(gè)Attribute是最重要的東西
public int UserId { get; set;}
public string UserName { get; set; }
public string Cellphone { get; set; }
}
public void Set<T>(T obj)
{
//此處應(yīng)該被緩存以提高反射的效率
var type = typeof(T);
var primaryKey = type.GetProperties()
.FirstOrDefault(t => t.GetCustomAttributes(false)
.Any(c => c is PrimaryKeyAttribute));//這里通過(guò)取PrimaryKeyAttribute來(lái)獲取ID的value
var keyValue = primaryKey.GetValue(obj, null);
var key = string.Format("urn:{0}:{1}", type.Name, keyValue);
var dt = DateTime.UtcNow.AddDays(1);//假設(shè)默認(rèn)緩存1天
var offset = new DateTimeOffset(dt);
Cache.Set(key, obj, offset);
}
到這里,我想到的最終版本的ICache就完成了。這里還需要說(shuō)明的是其實(shí)PrimaryKey可以更加靈活多變。很多時(shí)候一個(gè)Object的PrimaryKey是很復(fù)雜的,這時(shí)候設(shè)計(jì)Cache實(shí)體的時(shí)候可以變通下:
public class UserCacheEntity
{
[PrimaryKey]
public int ID
{
get
{
return string.Format("{0}:{1}", UserId, UserName);
}
}
public int UserId { get; set; }
public string UserName { get; set; }
public string Cellphone { get; set; }
}
上面的方式幾乎可以自動(dòng)管理常見的數(shù)據(jù)Cache了,唯一麻煩的是 需要自定義一個(gè)CacheObject,這樣就帶來(lái)了實(shí)體轉(zhuǎn)換的麻煩,這時(shí)候就要看怎么取舍了。
再次說(shuō)明下我想要的ICache設(shè)計(jì):
1. 永遠(yuǎn)只Cache熱數(shù)據(jù),這意味著每個(gè)Key都要有過(guò)期時(shí)間
2. ICache自動(dòng)管理Get/Set,最好能自動(dòng)管理Key。
3. ICache精簡(jiǎn)同時(shí)又不失靈活。
詳細(xì)的代碼Demo可以參考:Git
更靈活的實(shí)現(xiàn)
我在寫這篇總結(jié)之前,也一直在思考Cache應(yīng)該放到什么層,普通三層的時(shí)候放哪里?DDD那樣分層的時(shí)候又放哪里。Google了下,看到了一些參考。
http://stackoverflow.com/questions/15340173/in-which-layer-implement-the-cache
我覺得這里比較符合我的想法,Cache應(yīng)該是全局任意的,當(dāng)然實(shí)現(xiàn)起來(lái)當(dāng)然是interface+IOC,這樣引用起來(lái)更加的獨(dú)立一些。
另外還有Cache更加高級(jí)的使用,AOP結(jié)合ICache V4這樣的設(shè)計(jì),豈不是更好?這里我還沒(méi)有去實(shí)現(xiàn)AOP的Attribute,這又是一個(gè)大話題的,下次再來(lái)實(shí)現(xiàn)吧。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- ASP.NET緩存管理的幾種方法
- asp.net(C#)遍歷memcached緩存對(duì)象
- Asp.Net Cache緩存使用代碼
- ASP.NET網(wǎng)站管理系統(tǒng)退出 清除瀏覽器緩存,Session的代碼
- .net/c# memcached緩存獲取所有緩存鍵的方法步驟
- asp.net 客戶端瀏覽器緩存的Http頭介紹
- ASP.net Substitution 頁(yè)面緩存而部分不緩存的實(shí)現(xiàn)方法
- ASP.NET性能優(yōu)化之讓瀏覽器緩存動(dòng)態(tài)網(wǎng)頁(yè)的方法
- ASP.NET頁(yè)面在IE緩存的清除辦法
- asp.net 提高網(wǎng)站速度及如何利用緩存
相關(guān)文章
ASP.NET Core中快速構(gòu)建PDF文檔的步驟分享
這篇文章主要給大家介紹了關(guān)于ASP.NET Core中快速構(gòu)建PDF文檔的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用ASP.NET Core具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
輕量級(jí)ORM框架Dapper應(yīng)用之安裝Dapper
這篇文章介紹了輕量級(jí)ORM框架Dapper的安裝方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-03-03
用vs調(diào)試sql存儲(chǔ)過(guò)程圖文介紹
想必大家應(yīng)該有給存儲(chǔ)過(guò)程找錯(cuò)誤的經(jīng)歷吧,一遍遍的去讀sql代碼,一句一句的測(cè)試,發(fā)現(xiàn)一個(gè)小錯(cuò)誤可能都要用很長(zhǎng)的時(shí)間,接下來(lái)將介紹使用vs2010調(diào)試存儲(chǔ)過(guò)程,感興趣的朋友可以不要錯(cuò)過(guò)了啊2013-02-02
mstest實(shí)現(xiàn)類似單元測(cè)試nunit中assert.throws功能
我們做單元測(cè)試NUnit中,有一個(gè)斷言Assert.Throws很好用,現(xiàn)在我們來(lái)擴(kuò)展一下也實(shí)現(xiàn)類似成功能,大家參考使用吧2014-01-01
jQuery實(shí)現(xiàn)倒計(jì)時(shí)跳轉(zhuǎn)的例子
這篇文章主要介紹了jQuery實(shí)現(xiàn)倒計(jì)時(shí)跳轉(zhuǎn)的例子,需要的朋友可以參考下2014-05-05
關(guān)于ASP.NET頁(yè)面打印技術(shù)的常用方法總結(jié)
B/S結(jié)構(gòu)導(dǎo)致了Web應(yīng)用程序中打印的特殊性;程序運(yùn)行在瀏覽器中,打印機(jī)在本地,而文件確可能在服務(wù)器上,導(dǎo)致了打印控制不是很靈活,接下來(lái)介紹幾種常見的打印技術(shù),感興趣的朋友可以了解下2013-01-01
Asp.net ajax實(shí)現(xiàn)任務(wù)提示頁(yè)面的簡(jiǎn)單代碼
這篇文章介紹了Asp.net ajax實(shí)現(xiàn)任務(wù)提示頁(yè)面的簡(jiǎn)單代碼,有需要的朋友可以參考一下2013-11-11
DataSet.Tables[].Rows[][]的用法詳細(xì)解析
以下是對(duì)DataSet.Tables[].Rows[][]的用法進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過(guò)來(lái)參考下2013-09-09

