0

0

C# 文件系统的弹性 C#如何设计一个能够容忍磁盘故障的文件处理系统

畫卷琴夢

畫卷琴夢

发布时间:2026-02-27 11:39:10

|

364人浏览过

|

来源于php中文网

原创

file.copy磁盘故障需手动重试并判错:捕获ioexception,检查hresult为0x80070015等码,对设备未就绪指数退避重试3次;多磁盘场景用diskrouter前置选盘,懒检测健康状态;跨卷move须拆解为复制+manifest记录+删除,确保可回退;日志写入要autoflush或定时flush防缓冲掩盖故障。

c# 文件系统的弹性 c#如何设计一个能够容忍磁盘故障的文件处理系统

磁盘故障时 File.Copy 直接抛 IOException 怎么办

它不区分“磁盘暂时忙”和“磁盘已离线”,只要底层 CreateFile 失败,就立刻 throw。这不是设计缺陷,而是 Windows API 的默认行为——C# 只是透传了这个语义。

实操上必须自己兜底重试 + 判定错误类型:

  • 捕获 IOException 后检查 InnerException?.HResult 是否为 0x80070015(设备未就绪)、0x80070070(磁盘空间不足,但可能实为坏道假报)、或 0x80070020(进程正占用)
  • 0x80070015 做指数退避重试(比如 1s、2s、4s),最多 3 次;超过则认为磁盘已不可用
  • 避免在重试中反复调用 File.Exists——它本身也会触发磁盘访问,可能加重故障

单机多磁盘场景下如何让 FileStream 自动降级到备用盘

不能靠 try-catch 后手动切路径,因为 FileStream 构造即打开句柄,失败就结束了。得把“选择哪块盘”从 IO 路径里抽出来,变成前置决策。

推荐做法是封装一个 DiskRouter 类,内部维护磁盘健康状态缓存:

  • 启动时用 DriveInfo.GetDrives() 扫描所有本地固定磁盘,逐个执行 new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.None, 1, FileOptions.RandomAccess) 测试可写性(只开不读写,快速验证)
  • 将健康盘按剩余空间排序,故障盘标记为 Unhealthy 并记录最后失败时间
  • 业务调用 DiskRouter.GetWritablePath("myapp/logs") 时,它返回当前最优路径,而非硬编码盘符

注意:不要每秒轮询磁盘状态,改用“首次访问失败 + 延迟 30 秒后重新探测”的懒检测策略,减少干扰。

Getsound
Getsound

基于当前天气条件生成个性化音景音乐

下载

Directory.Move 跨卷失败时,如何无损回退到复制+删除

跨卷移动本质是复制+删除,但 Directory.Move 把这两步锁死在原子操作里,一旦中间出错(比如目标盘突然满),源目录可能已部分清空,无法恢复。

必须拆解成可控步骤:

  • 先用 Directory.EnumerateFiles(source, "*", SearchOption.AllDirectories) 获取全量文件路径快照(内存小,不占磁盘)
  • 逐个 File.Copy 到目标,每成功一个就记录到临时 .move_manifest.json 文件(写入目标盘根目录)
  • 全部复制完成后,再执行源目录递归删除;若删除中途失败,根据 manifest 反向清理目标盘残留文件
  • 关键点:.move_manifest.json 必须用 File.WriteAllText(path, json, Encoding.UTF8) + FileOptions.WriteThrough 确保落盘,否则断电时 manifest 丢失,回退失效

日志写入卡住时,StreamWriter 缓冲区会掩盖磁盘故障

默认 StreamWriter 开启缓冲,即使磁盘已掉线,WriteLine 仍返回成功,直到缓冲区满或 Flush 时才暴露问题——这会让故障感知延迟几十秒甚至更久。

生产环境必须关闭自动缓冲或强制同步:

  • 构造时传 new StreamWriter(stream) { AutoFlush = true },但注意性能损耗(每次写都落盘)
  • 更平衡的做法:保持缓冲,但每 5 秒调用一次 streamWriter.Flush(),并在 catch (IOException) 后立即尝试 stream.Flush() 验证底层是否存活
  • 绝对不要依赖 Dispose 触发的隐式 Flush——进程崩溃时它根本不会执行

真正难处理的是“磁盘响应极慢但不死”,这时候需要单独起一个 watchdog 线程,用 stream.BeginWrite + 超时判断,而不是等同步写卡住整个主线程。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

449

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

326

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

81

2025.09.10

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

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

721

2023.08.10

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

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

721

2023.08.10

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

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

721

2023.08.10

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

1312

2023.07.26

html5播放器怎么用
html5播放器怎么用

本合集全面介绍HTML5播放器的使用方法,涵盖基础语法、自定义控制、兼容性处理及实战示例。阅读专题下面的文章了解更多详细内容。

0

2026.02.27

热门下载

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

精品课程

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

共94课时 | 10.3万人学习

C 教程
C 教程

共75课时 | 5万人学习

C++教程
C++教程

共115课时 | 19.6万人学习

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

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