.Net彈性和瞬態(tài)故障處理庫(kù)Polly實(shí)現(xiàn)彈性策略
和故障處理策略不同的是,彈性策略并不是針對(duì)委托執(zhí)行過(guò)程中的異常進(jìn)行處理,而是改變委托本身的行為,因此彈性策略并沒(méi)有故障定義這一過(guò)程,它的處理流程為:
- 定義策略
- 應(yīng)用策略
Polly對(duì)彈性策略也做了不少支持,本文這里就簡(jiǎn)單的介紹一下。
彈性策略:超時(shí)(Timeout)
超時(shí)策略用于控制委托的運(yùn)行時(shí)間,如果達(dá)到指定時(shí)間還沒(méi)有運(yùn)行,則觸發(fā)超時(shí)異常。
Policy.Timeout(TimeSpan.FromSeconds(3), TimeoutStrategy.Pessimistic);
超時(shí)策略常見(jiàn)的重載版本有如下幾個(gè):
Policy.Timeout(300); Policy.Timeout(TimeSpan.FromMilliseconds(3)); Policy.Timeout(() => TimeSpan.FromSeconds(3)); Policy.Timeout(TimeSpan.FromSeconds(3), TimeoutStrategy.Optimistic);
超時(shí)策略:
Polly支持兩種超時(shí)策略:
- TimeoutStrategy.Pessimistic: 悲觀模式
當(dāng)委托到達(dá)指定時(shí)間沒(méi)有返回時(shí),不繼續(xù)等待委托完成,并拋超時(shí)TimeoutRejectedException異常。 - TimeoutStrategy.Optimistic:樂(lè)觀模式
這個(gè)模式依賴于 co-operative cancellation,只是觸發(fā)CancellationTokenSource.Cancel函數(shù),需要等待委托自行終止操作。
其中悲觀模式比較容易使用,因?yàn)樗恍枰谖蓄~外的操作,但由于它本身無(wú)法控制委托的運(yùn)行,函數(shù)本身并不知道自己被外圍策略取消了,也無(wú)法在超時(shí)的時(shí)候中斷后續(xù)行為。因此用起來(lái)反而還不是那么實(shí)用。
一個(gè)樂(lè)觀模式的的策略示例如下:
var policy = Policy.Timeout(300); var cts = new CancellationTokenSource(); policy.Execute(ct => { for (int i = 0; i < 1000; i++) { Thread.Sleep(100); ct.ThrowIfCancellationRequested(); } }, cts.Token);
復(fù)合策略:
在日常的使用中,僅僅只有超時(shí)往往是并不夠的,很多時(shí)候還需要和重試等其它故障處理策略一起使用,如:
Policy.Handle<TimeoutRejectedException>() .Retry(3) .Wrap(Policy.Timeout(3, TimeoutStrategy.Pessimistic));
彈性策略:無(wú)操作(NoOp)
在開(kāi)發(fā)過(guò)程中,處于測(cè)試或定位問(wèn)題時(shí)的需要,有時(shí)我們也需要一個(gè)沒(méi)有任何行為的策略,Polly系統(tǒng)默認(rèn)提供了一個(gè).
Policy.NoOp();
這個(gè)啥都沒(méi)干,也沒(méi)啥好介紹的了。
彈性策略:緩存(Cache)
有的時(shí)候,數(shù)據(jù)更新并不是頻繁的,此時(shí)可以使用緩存策略減少對(duì)服務(wù)的訪問(wèn),提高系統(tǒng)性能:
var memoryCacheProvider = new Polly.Caching.MemoryCache.MemoryCacheProvider(new System.Runtime.Caching.MemoryCache("cache")); var cachePolicy = Policy.Cache(memoryCacheProvider, TimeSpan.FromMinutes(5)); //Context.ExecutionKey就是cache的key var context = new Context("cache_key"); for (int i = 0; i < 3; i++) { var cache = cachePolicy.Execute(_ => { Console.WriteLine("get value"); return 3; }, context); Console.WriteLine(cache); }
PS:這個(gè)示例使用了MemoryCache,需要使用Nuget安裝Polly.Caching.MemoryCache程序包,以及添加System.Runtime.Caching的引用。
從運(yùn)行結(jié)果可以看到,雖然三次執(zhí)行都有結(jié)果,但系統(tǒng)只有第一次才需要執(zhí)行函數(shù),剩下兩次都是直接從緩存中獲取的結(jié)果。
系統(tǒng)也提供了多種不同的過(guò)期策略:
Policy.Cache(memoryCacheProvider, new AbsoluteTtl(DateTimeOffset.Now.Date.AddDays(1))); Policy.Cache(memoryCacheProvider, new SlidingTtl(TimeSpan.FromMinutes(5)));
對(duì)于布式緩存,Polly也有默認(rèn)的實(shí)現(xiàn),只需要安裝Polly.Caching.IdistributedCache程序包即可,它提供了SqlServer和Redis的支持。
關(guān)于Cache的更多內(nèi)容,可以參考官方文檔:Cache
彈性策略:艙壁隔離(Bulkhead Isolation)
艙壁隔離是一種并發(fā)控制的行為,并發(fā)控制是一個(gè)比較常見(jiàn)的模式,Polly也提供了這方面的支持,如:
//該策略下最多只有12個(gè)任務(wù)并發(fā)執(zhí)行 Policy.Bulkhead(12);
超過(guò)了并發(fā)數(shù)的任務(wù)會(huì)拋BulkheadRejectedException,如果要放在隊(duì)列中等待,Polly也提供了等待隊(duì)列的支持:
Policy.Bulkhead(12, 100);
這種方式下,有12個(gè)并發(fā)任務(wù),每個(gè)任務(wù)維持著一個(gè)并發(fā)隊(duì)列,每個(gè)隊(duì)列可以自持最大100個(gè)任務(wù)。
不過(guò),和微軟自己的DataFlow模塊比起來(lái),感覺(jué)Polly模塊的并發(fā)控制的功能還是比較弱的。不過(guò)這也它本身的應(yīng)用場(chǎng)景也相關(guān),如果需要更強(qiáng)大的策略,也可以自行封裝。
彈性策略:策略封裝(PolicyWrap)
我們可以通過(guò)PolicyWrap的方式,封裝出一個(gè)更加強(qiáng)大的策略:
var fallback = Policy<int>.Handle<TimeoutException>().Fallback(100); var retry = Policy<int>.Handle<TimeoutException>().Retry(2); Policy.Wrap(fallback, retry);
這個(gè)策略就是將Retry和Fallback組合起來(lái),形成一個(gè)retry and fallback的策略,也可以寫(xiě)成如下形式:
var retryAndFallback = fallback.Wrap(retry);
當(dāng)執(zhí)行這個(gè)新策略時(shí):
retryAndFallback.Execute(DoSomething);
等價(jià)于執(zhí)行:
fallback.Execute(()=> retry.Execute(DoSomething));
到此這篇關(guān)于Polly實(shí)現(xiàn)彈性策略的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Asp.Net Core利用xUnit進(jìn)行主機(jī)級(jí)別的網(wǎng)絡(luò)集成測(cè)試詳解
這篇文章主要給大家介紹了關(guān)于Asp.Net Core利用xUnit進(jìn)行主機(jī)級(jí)別的網(wǎng)絡(luò)集成測(cè)試的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們來(lái)一起看看吧2018-12-12aspx實(shí)現(xiàn)的 jquery ui 的 flexgrid demo
這幾天沒(méi)事研究著jquery,真是個(gè)好東西,慢慢的知道了有jquery ui,一開(kāi)始就被華麗的界面和簡(jiǎn)單的操作給吸引了,尤其是里面的flexgrid,對(duì)我而言可以說(shuō)是非常寶貴的東西2009-12-12Asp.net?core?使用SignalR推送消息過(guò)程詳解
ASP.NET?Core?SignalR?是一個(gè)開(kāi)放源代碼庫(kù),可用于簡(jiǎn)化向應(yīng)用添加實(shí)時(shí)?Web?功能。?實(shí)時(shí)?Web?功能使服務(wù)器端代碼能夠?qū)?nèi)容推送到客戶端,本文重點(diǎn)給大家介紹Asp.net?core?使用SignalR推送消息,感興趣的朋友一起看看吧2022-03-03asp.net DataTable相關(guān)操作集錦(篩選,取前N條數(shù)據(jù),去重復(fù)行,獲取指定列數(shù)據(jù)等)
這篇文章主要介紹了asp.net DataTable相關(guān)操作,包括篩選,取前N條數(shù)據(jù),去重復(fù)行,獲取指定列數(shù)據(jù)等.基本涵蓋了DataTable的常見(jiàn)操作技巧,需要的朋友可以參考下2016-06-06ASP.NET中控件的EnableViewState屬性及徹底禁用
如果我們?cè)陂_(kāi)發(fā)Web應(yīng)用程序時(shí),某些控件是不需要接受用戶的操作或只需要接受一次操作的時(shí)候,我們可以將這些控件的EnableViewState屬性改為false,這樣可以優(yōu)化我們的程序,提高網(wǎng)絡(luò)訪問(wèn)的速度。2016-06-06asp.net 水晶報(bào)表隔行換色實(shí)現(xiàn)方法
水晶報(bào)表隔行換色實(shí)現(xiàn)方法,需要的朋友可以參考下。2009-11-11