0

0

c# Polly的重试(Retry)和回退(Fallback)策略怎么用

幻夢星雲

幻夢星雲

发布时间:2026-01-10 09:09:35

|

800人浏览过

|

来源于php中文网

原创

Retry策略需显式指定异常或结果处理条件,推荐指数退避;Fallback提供兜底行为,须与Retry类型一致并组合使用;异步场景下必须全程传递CancellationToken。

c# polly的重试(retry)和回退(fallback)策略怎么用

Retry 策略:什么时候重试、重试几次、间隔怎么设

Retry 不是盲目重试,它只对特定异常或返回值生效。默认情况下 PollyRetry 不捕获任何异常,必须显式指定 Handle 或用谓词判断。

常见错误是直接写 WaitAndRetry(3) 却没指定捕获哪些异常,结果异常直接抛出,重试根本没触发。

  • Handle() 捕获 HTTP 请求失败
  • HandleResult(r => r == null || r.IsSuccess == false) 处理返回值语义失败(比如 API 返回 200 但 body 中 success: false
  • 推荐用指数退避:WaitAndRetryAsync(3, retryAttempt => TimeSpan.FromSeconds(Math.Pow(2, retryAttempt))),避免雪崩
  • 注意:同步 Retry(如 Execute)会阻塞线程;I/O 场景务必用 RetryAsync + ExecuteAsync
var retryPolicy = Policy
    .Handle()
    .OrResult(r => !r.IsSuccessStatusCode)
    .WaitAndRetryAsync(
        retryCount: 3,
        sleepDurationProvider: retryAttempt => TimeSpan.FromMilliseconds(100 * Math.Pow(2, retryAttempt)),
        onRetry: (outcome, timespan, retryCount, context) =>
        {
            Console.WriteLine($"Retry {retryCount} after {timespan.TotalMilliseconds}ms");
        });

Fallback 策略:当重试也失败时,提供兜底行为

Fallback 不是“重试失败后自动执行”,而是独立策略,常与 Retry 组合使用(通过 WrapAsync)。它的核心是定义“失败时返回什么”或“执行什么补偿逻辑”。

容易踩的坑:把 Fallback 当作日志记录或监控钩子单独用——它必须有明确的返回值(或 Task),否则编译不通过;且一旦触发,原始异常/结果就丢弃了。

  • 返回默认值:FallbackAsync("default"),适用于 string 类型操作
  • 执行异步补偿逻辑:FallbackAsync(async ct => { await LogFailureAsync(); return default(T); })
  • 访问原始异常:用 onFallbackAsync 回调,参数是 DelegateResult,其中 Outcome.ExceptionOutcome.Result 可取到上下文
  • 注意:Fallback 的泛型类型必须和被包装的策略一致,否则 WrapAsync 编译失败
var fallbackPolicy = Policy
    .Handle()
    .OrResult(r => r == null)
    .FallbackAsync(
        fallbackValue: "fallback-content",
        onFallbackAsync: (outcome, context) =>
        {
            var ex = outcome.Exception?.InnerException ?? outcome.Exception;
            Console.WriteLine($"Fallback triggered: {ex?.Message}");
            return Task.CompletedTask;
        });

Retry + Fallback 组合:顺序和异常传递很关键

Policy.WrapAsync(retryPolicy, fallbackPolicy) 时,执行流是:先走 Retry,所有重试耗尽后仍失败 → 触发 Fallback。但 Fallback 并不会“看到” Retry 过程中的每一次失败,只看到最终失败结果。

Sider
Sider

多功能AI浏览器助手,帮助用户进行聊天、写作、阅读、翻译等

下载

真正容易忽略的是上下文传递和异常屏蔽:如果 Retry 内部已处理并吞掉异常(比如用 ExecuteAndCaptureAsync),Fallback 就收不到异常;反之,Fallback 若抛出新异常,上层就收不到原始异常信息。

  • 组合前确认两个策略的泛型类型严格一致(比如都是 Policy
  • 调试时在 onRetryonFallbackAsync 中打印日志,验证是否按预期触发
  • 不要在 Fallback 里再调用可能失败的外部服务——这会让兜底逻辑本身不可靠
  • 若需区分“重试全部失败”和“首次就失败”,可在 context 中传入标记,或在 onFallbackAsync 中检查 outcome.FinalException

异步策略下 cancellation token 的传递不能漏

所有 *Async 方法都接受 CancellationToken,但很多人只传给最外层 ExecuteAsync,忘了策略内部也需要它。比如网络请求超时后,Retry 的等待延时仍会继续执行,造成“取消不彻底”。

WaitAndRetryAsyncsleepDurationProvider 函数本身不接收 CancellationToken,但你可以在回调里主动检查;更稳妥的做法是用 WaitAndRetryAsync(..., onRetryAsync: ...) 的异步版本,在其中 ct.ThrowIfCancellationRequested()

  • 始终把 cancellationToken 传给 ExecuteAsync(..., cancellationToken)
  • onRetryAsync 回调开头加 ct.ThrowIfCancellationRequested()
  • Fallback 的 fallbackAction 如果是异步委托,也必须声明 CancellationToken 参数并参与协作取消

组合策略的健壮性不取决于重试次数多寡,而在于每次失败是否被准确识别、每次等待是否可中断、每次兜底是否真正可控——这些细节在压测或网络抖动时才会暴露。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

435

2024.03.01

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6073

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

795

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1056

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

1199

2024.03.01

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

478

2023.08.10

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号