0

0

C# Polly文件操作重试 C#如何为不稳定的文件IO增加重试策略

幻夢星雲

幻夢星雲

发布时间:2026-02-26 11:27:12

|

423人浏览过

|

来源于php中文网

原创

c# polly文件操作重试 c#如何为不稳定的文件io增加重试策略

为什么直接用 Polly.Wrap 会失败

文件 IO 操作(比如 File.ReadAllTextFileStream.Write)通常抛出的是 IOExceptionUnauthorizedAccessExceptionDirectoryNotFoundException,但 Polly 默认策略只捕获 Exception 基类——这本身没问题;真正踩坑的是:**某些文件异常(如被其他进程锁定)在重试瞬间仍会立即复现,且没有“冷却时间”或“退避逻辑”,导致重试形同虚设**。

更隐蔽的问题是:Polly 的 PolicyWrap 如果嵌套了 RetryCircuitBreaker,而你没在重试前释放文件句柄(比如忘了 usingDispose),下次重试时可能因句柄未释放继续报 IOException: The process cannot access the file...

实操建议:

  • 必须用 Policy.Handle<ioexception>().Or<unauthorizedaccessexception>()</unauthorizedaccessexception></ioexception> 显式声明要捕获的异常类型,避免漏掉常见文件锁异常
  • 重试策略必须搭配 WaitAndRetryAsync + 指数退避(如 Backoff.DecorrelatedJitterBackoffV2),不能只用固定间隔
  • 所有文件操作必须包裹在 using 或确保 Dispose 被调用,否则重试只会让问题恶化

如何写一个安全的重试型 File.ReadAllText

直接对 File.ReadAllText(path) 套 Polly 是危险的——它内部可能已打开文件但没暴露流供你控制生命周期。正确做法是自己构造可重试的流读取逻辑。

示例(使用 FileStream + StreamReader):

var retryPolicy = Policy
    .Handle<IOException>()
    .Or<UnauthorizedAccessException>()
    .WaitAndRetryAsync(
        retryCount: 3,
        sleepDurationProvider: (retryAttempt) => 
            TimeSpan.FromMilliseconds(100 * Math.Pow(2, retryAttempt))
    );
<p>string content = await retryPolicy.ExecuteAsync(async () =>
{
using var fs = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read, 4096, FileOptions.Asynchronous);
using var sr = new StreamReader(fs);
return await sr.ReadToEndAsync();
});

关键点:

  • FileStream 创建和 StreamReader 初始化都放在 ExecuteAsync 内部,确保每次重试都是全新句柄
  • 显式指定 FileOptions.Asynchronous,避免同步阻塞干扰重试调度
  • 缓冲区大小(4096)建议设为 4KB 或 8KB,太小会放大 I/O 开销,太大无益

Directory.CreateDirectory 重试要注意什么

Directory.CreateDirectory 看似幂等,但在网络路径或权限瞬变场景下仍可能失败。它的典型错误是 UnauthorizedAccessExceptionIOException(如父目录不存在且无法创建)。

Descript
Descript

一个多功能的音频和视频编辑引擎

下载

但它有个隐藏特性:**即使抛异常,部分中间目录可能已被创建成功**。所以重试前不能简单“再试一次”,得先检查目标路径是否存在。

实操建议:

  • 重试策略应包含 .CanRetryImmediately((ex, ct) => !Directory.Exists(path)) 判断(需自定义策略逻辑,或手动加前置检查)
  • 不要对整个路径链盲目重试;优先确认最深层缺失的父目录,再逐级创建
  • 若用于 UNC 路径(如 \servershare older),必须确保运行账户有网络访问权限,Polly 无法绕过这个限制

异步文件写入(WriteAllTextAsync)怎么加重试

File.WriteAllTextAsync 底层会创建新文件并覆盖,但如果目标文件正被记事本、Excel 或杀毒软件占用,就会抛 IOException。注意:它不会自动处理“文件存在但只读”的情况——那会抛 UnauthorizedAccessException,需额外捕获。

推荐写法(带存在性与只读处理):

var policy = Policy
    .Handle<IOException>()
    .Or<UnauthorizedAccessException>()
    .WaitAndRetryAsync(2, _ => TimeSpan.FromMilliseconds(50));
<p>await policy.ExecuteAsync(async () =>
{
if (File.Exists(path) && (File.GetAttributes(path) & FileAttributes.ReadOnly) == FileAttributes.ReadOnly)
{
File.SetAttributes(path, FileAttributes.Normal);
}
await File.WriteAllTextAsync(path, content);
});

容易忽略的点:

  • File.SetAttributes 本身也可能失败(如权限不足),所以它应该放在重试范围内,而不是前置一次性操作
  • 如果写入内容很大,考虑改用 FileStream 分块写 + FlushAsync,避免单次大写入失败后全部重放
  • 不要在重试中重复生成临时文件名——比如用 Path.GetTempFileName(),否则每次重试都会留垃圾文件

重试不是万能胶水,文件 IO 的不确定性主要来自外部系统(OS 锁、防病毒扫描、网络延迟)。最可靠的策略永远是:先检查再操作,失败后等一会儿再检查,而不是无脑重试

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

329

2023.10.09

vb连接access数据库的方法
vb连接access数据库的方法

vb连接access数据库方法:1、使用ADO连接,首先导入System.Data.OleDb模块,然后定义一个连接字符串,接着创建一个OleDbConnection对象并使用Open() 方法打开连接;2、使用DAO连接,首先导入 Microsoft.Jet.OLEDB模块,然后定义一个连接字符串,接着创建一个JetConnection对象并使用Open()方法打开连接即可。

452

2023.10.16

asp连接access数据库的方法
asp连接access数据库的方法

连接的方法:1、使用ADO连接数据库;2、使用DSN连接数据库;3、使用连接字符串连接数据库。想了解更详细的asp连接access数据库的方法,可以阅读本专题下面的文章。

123

2023.10.18

access和trunk端口的区别
access和trunk端口的区别

access和trunk端口的区别是Access端口用于连接终端设备,提供单个VLAN的接入,而Trunk端口用于连接交换机之间,提供多个VLAN的传输;Access端口只传输属于指定VLAN的数据,而Trunk端口可以传输多个VLAN的数据,并使用VLAN标签进行区分。想了解更多access和trunk端口相关内容,可以阅读本专题下面的文章。

336

2023.10.31

access怎么导入数据
access怎么导入数据

access导入数据步骤:1. 选择数据源 2. 选择要导入的文件 3. 指定导入选项 4. 选择导入目标 5. 预览数据 6. 导入数据即可。想了解更多access的相关内容,可以阅读本专题下面的文章。

452

2024.04.10

access数据库用途
access数据库用途

access数据库是一种关系型数据库管理系统,主要用途包括:数据存储和管理;数据查询和检索;报告和表单设计;应用程序开发。想了解更多access数据库的相关内容,可以阅读本专题下面的文章。

590

2024.04.10

excel对比两列数据异同
excel对比两列数据异同

Excel作为数据的小型载体,在日常工作中经常会遇到需要核对两列数据的情况,本专题为大家提供excel对比两列数据异同相关的文章,大家可以免费体验。

1444

2023.07.25

excel重复项筛选标色
excel重复项筛选标色

excel的重复项筛选标色功能使我们能够快速找到和处理数据中的重复值。本专题为大家提供excel重复项筛选标色的相关的文章、下载、课程内容,供大家免费下载体验。

426

2023.07.31

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

331

2026.02.25

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.2万人学习

C 教程
C 教程

共75课时 | 5万人学习

C++教程
C++教程

共115课时 | 19.5万人学习

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

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