C#調(diào)用Polly庫(kù)實(shí)現(xiàn)捕捉異常處理的操作代碼
一、Polly介紹
官方對(duì) Polly 的介紹是這樣的:
Polly is a .NET resilience and transient-fault-handling library that allows developers to express policies such as Retry, Circuit Breaker, Timeout, Bulkhead Isolation, and Fallback in a fluent and thread-safe manner.
翻譯過(guò)來(lái)大概意思是:Polly 是一個(gè) .NET 彈性和瞬態(tài)故障處理庫(kù),允許開(kāi)發(fā)人員以 Fluent 和線程安全的方式來(lái)實(shí)現(xiàn)重試、斷路、超時(shí)、隔離和回退策略。
二、Polly 的七種策略
Polly 可以實(shí)現(xiàn)重試、斷路、超時(shí)、隔離、回退和緩存策略
1、重試(Retry)
出現(xiàn)故障自動(dòng)重試
//捕捉到錯(cuò)誤,自動(dòng)重試三次 Policy.Handle<Exception>() .Retry(3, (exception, retryCount) => { Console.WriteLine(DateTime.Now + $"代碼異常: {exception},開(kāi)始第 {retryCount} 次重試。"); });
2、 斷路(Circuit-breaker)
當(dāng)系統(tǒng)遇到嚴(yán)重問(wèn)題時(shí),快速回饋失敗比讓用戶/調(diào)用者等待要好,限制系統(tǒng)出錯(cuò)的體量,有助于系統(tǒng)恢復(fù)。比如,當(dāng)我們?nèi)フ{(diào)一個(gè)第三方的 API,有很長(zhǎng)一段時(shí)間 API 都沒(méi)有響應(yīng),可能對(duì)方服務(wù)器癱瘓了。如果我們的系統(tǒng)還不停地重試,不僅會(huì)加重系統(tǒng)的負(fù)擔(dān),還會(huì)可能導(dǎo)致系統(tǒng)其它任務(wù)受影響。所以,當(dāng)系統(tǒng)出錯(cuò)的次數(shù)超過(guò)了指定的閾值,就要中斷當(dāng)前線路,等待一段時(shí)間后再繼續(xù)。
//捕捉到同一個(gè)異常兩次時(shí),就停下來(lái),等待 1 分鐘后再繼續(xù) Policy.Handle<SomeException>() .CircuitBreaker(2, TimeSpan.FromMinutes(1));
3、 超時(shí)(Timeout)
當(dāng)系統(tǒng)超過(guò)一定時(shí)間的等待,我們就幾乎可以判斷不可能會(huì)有成功的結(jié)果。比如平時(shí)一個(gè)網(wǎng)絡(luò)請(qǐng)求瞬間就完成了,如果有一次網(wǎng)絡(luò)請(qǐng)求超過(guò)了 30 秒還沒(méi)完成,我們就知道這次大概率是不會(huì)返回成功的結(jié)果了。因此,我們需要設(shè)置系統(tǒng)的超時(shí)時(shí)間,避免系統(tǒng)長(zhǎng)時(shí)間做無(wú)謂的等待。
//這里設(shè)置了超時(shí)時(shí)間不能超過(guò) 30 秒,否則就認(rèn)為是錯(cuò)誤的結(jié)果,并執(zhí)行回調(diào) Policy.Timeout(30, onTimeout: (context, timespan, task) => { Console.WriteLine("30秒無(wú)響應(yīng),需要執(zhí)行的邏輯"); });
4、 隔離(Bulkhead Isolation)
當(dāng)系統(tǒng)的一處出現(xiàn)故障時(shí),可能促發(fā)多個(gè)失敗的調(diào)用,很容易耗盡主機(jī)的資源(如 CPU)。下游系統(tǒng)出現(xiàn)故障可能導(dǎo)致上游的故障的調(diào)用,甚至可能蔓延到導(dǎo)致系統(tǒng)崩潰。所以要將可控的操作限制在一個(gè)固定大小的資源池中,以隔離有潛在可能相互影響的操作。
//這個(gè)策略是最多允許 10 個(gè)線程并發(fā)執(zhí)行,如果執(zhí)行被拒絕,則執(zhí)行回調(diào) Policy.Bulkhead(10, context => { Console.WriteLine("當(dāng)前線程超過(guò)10個(gè),需要執(zhí)行的邏輯"); });
5、 回退(Fallback)
有些錯(cuò)誤無(wú)法避免,就要有備用的方案。這個(gè)就像瀏覽器不支持一些新的 CSS 特性就要額外引用一個(gè) polyfill 一樣。一般情況,當(dāng)無(wú)法避免的錯(cuò)誤發(fā)生時(shí),我們要有一個(gè)合理的返回來(lái)代替失敗。
//當(dāng)用戶沒(méi)有填寫(xiě)名稱(chēng)時(shí),我們就給他一個(gè)默認(rèn)名稱(chēng),這種策略可以這樣定義: Policy.Handle<Whatever>() .Fallback<User>(() => User.GetDefaultName())
6、 緩存(Cache)
一般我們會(huì)把頻繁使用且不會(huì)怎么變化的資源緩存起來(lái),以提高系統(tǒng)的響應(yīng)速度。如果不對(duì)緩存資源的調(diào)用進(jìn)行封裝,那么我們調(diào)用的時(shí)候就要先判斷緩存中有沒(méi)有這個(gè)資源,有的話就從緩存返回,否則就從資源存儲(chǔ)的地方(比如數(shù)據(jù)庫(kù))獲取后緩存起來(lái),再返回,而且有時(shí)還要考慮緩存過(guò)期和如何更新緩存的問(wèn)題。Polly 提供了緩存策略的支持,使得問(wèn)題變得簡(jiǎn)單。
//這是官方的一個(gè)使用示例用法, //它定義了緩存 5 分鐘過(guò)期的策略,然后把這個(gè)策略應(yīng)用在指定的 Key(即 `FooKey`)上。 var memoryCacheProvider = new MemoryCacheProvider(myMemoryCache); var cachePolicy = Policy.Cache(memoryCacheProvider, TimeSpan.FromMinutes(5)); TResult result = cachePolicy.Execute(context => getFoo(), new Context("FooKey"));
7、 策略包(Policy Wrap)
一種操作會(huì)有多種不同的故障,而不同的故障處理需要不同的策略。這些不同的策略必須包在一起,作為一個(gè)策略包,才能應(yīng)用在同一種操作上。也就是 Polly 的彈性,即各種不同的策略能夠靈活地組合起來(lái)。
//先是把預(yù)先定義好的多種不同的策略包在一起,作為一個(gè)整體策略,然后應(yīng)用在同一個(gè)操作上。 var policyWrap = Policy .Wrap(fallback, cache, retry, breaker, timeout, bulkhead); policyWrap.Execute(() => { Console.WriteLine("多種策略共同執(zhí)行的邏輯"); });
三、簡(jiǎn)單代碼示例
備注:先引入Nuget包(Polly)
//創(chuàng)建一個(gè)策略,用于處理異常,并進(jìn)行重試 public static Policy policy = Policy.Handle<Exception>() .Retry(3, (exception, retryCount) => { Console.WriteLine(DateTime.Now + $"代碼異常: {exception},開(kāi)始第 {retryCount} 次重試。"); }); public async Task Execute() { //重試機(jī)制 policy.Execute(() => //這里接收一個(gè)委托方法,也就是捕捉到異常之后要執(zhí)行的業(yè)務(wù)邏輯 { Console.WriteLine("重試"); }); }
以上就是C#調(diào)用Polly庫(kù)實(shí)現(xiàn)捕捉異常處理的操作代碼的詳細(xì)內(nèi)容,更多關(guān)于C# Polly庫(kù)捕捉異常的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#學(xué)習(xí)筆記整理_變量等基礎(chǔ)語(yǔ)法(必看篇)
下面小編就為大家?guī)?lái)一篇C#學(xué)習(xí)筆記整理_變量等基礎(chǔ)語(yǔ)法(必看篇)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-09-09Untiy Shader實(shí)現(xiàn)紋理貼圖滾動(dòng)
這篇文章主要為大家詳細(xì)介紹了Untiy Shader實(shí)現(xiàn)紋理貼圖滾動(dòng),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-03-03C#實(shí)現(xiàn)多個(gè)計(jì)時(shí)器記錄不同定時(shí)時(shí)間
這篇文章主要為大家詳細(xì)介紹了C#實(shí)現(xiàn)多個(gè)計(jì)時(shí)器記錄不同定時(shí)時(shí)間,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12C#實(shí)現(xiàn)動(dòng)態(tài)圖標(biāo)閃爍顯示的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用C#實(shí)現(xiàn)動(dòng)態(tài)圖標(biāo)閃爍顯示的功能,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C#有一定的幫助,感興趣的小伙伴可以了解一下2022-12-12C#基于面向過(guò)程計(jì)算加權(quán)平均分的方法
這篇文章主要介紹了C#基于面向過(guò)程計(jì)算加權(quán)平均分的方法,涉及C#數(shù)學(xué)運(yùn)算的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07