‌C# I/O 核心用法

news2025/3/21 21:26:12

在 C# 中,输入输出(I/O)操作是处理文件、目录、流等数据交互的核心功能。本文将从基础到高级,系统讲解 C# 中文件 I/O 的实现方式、最佳实践及常见场景解决方案。

一、核心类与命名空间‌

1‌、System.IO 命名空间‌,包含以下关键类:

  • File:静态方法实现文件的创建、读写、删除等。
  • Directory:操作目录(创建、遍历、删除)。
  • Path:安全处理文件路径(跨平台兼容)。
  • FileStream:提供字节级别的文件读写。
  • StreamReader/StreamWriter:封装流,支持文本编码处理。
  • FileInfo/DirectoryInfo:实例方法操作文件或目录属性。

2、其他相关命名空间‌

  • System.IO.Compression:处理 ZIP 压缩文件(如 ZipFile 类)。
  • System.Text.Encoding:定义文本编码(如 UTF-8、GB2312)。

‌二、文件读写操作‌

‌1. 文本文件读写‌

  • ‌快速读写(适合小文件)‌
    使用 File 类的静态方法:
// 读取全部内容
string content = File.ReadAllText("file.txt", Encoding.UTF8);

// 写入内容(覆盖)
File.WriteAllText("file.txt", "Hello World");

// 追加内容
File.AppendAllText("file.txt", "\nNew Line");
  • 流式读写(适合大文件或逐行处理)‌
    使用 StreamReader 和 StreamWriter:
// 逐行读取大文件
using (var reader = new StreamReader("largefile.txt"))
{
    string line;
    while ((line = reader.ReadLine()) != null)
    {
        Console.WriteLine(line);
    }
}

// 流式写入
using (var writer = new StreamWriter("output.txt", append: true))
{
    writer.WriteLine("Line 1");
    writer.WriteLine("Line 2");
}

‌2. 二进制文件读写‌
使用 FileStream 直接操作字节:

// 写入二进制数据
byte[] data = new byte[] { 0x01, 0x02, 0x03 };
using (var stream = new FileStream("data.bin", FileMode.Create))
{
    stream.Write(data, 0, data.Length);
}

// 读取二进制数据
byte[] buffer = new byte[1024];
using (var stream = new FileStream("data.bin", FileMode.Open))
{
    int bytesRead = stream.Read(buffer, 0, buffer.Length);
}

‌三、流(Stream)的核心概念‌

‌1. 流的分类‌

  • ‌FileStream‌:处理文件流。
  • ‌MemoryStream‌:在内存中操作数据。
  • ‌NetworkStream‌:网络数据传输(如 TCP/IP)。

‌2. 流的通用操作‌

using (FileStream fs = new FileStream("file.txt", FileMode.Open, FileAccess.Read))
{
    // 设置流位置(支持随机访问)
    fs.Seek(10, SeekOrigin.Begin);

    // 读取字节
    byte[] buffer = new byte[1024];
    int bytesRead = fs.Read(buffer, 0, buffer.Length);
}

‌3. 缓冲区的优化‌

  • 通过指定缓冲区大小提升性能:
using (var fs = new FileStream("file.txt", FileMode.Open, FileAccess.Read, FileShare.Read, bufferSize: 4096))
{
    // 更高效的读写操作
}

‌四、异步 I/O 操作‌

  • 使用 async/await 避免阻塞主线程:
// 异步读取文件
public async Task<string> ReadFileAsync(string path)
{
    using (var reader = new StreamReader(path))
    {
        return await reader.ReadToEndAsync();
    }
}

// 异步写入文件
public async Task WriteFileAsync(string path, string content)
{
    using (var writer = new StreamWriter(path))
    {
        await writer.WriteAsync(content);
    }
}

‌五、目录与路径操作‌

‌1. 目录管理‌

// 创建目录(自动忽略已存在)
Directory.CreateDirectory(@"C:\data\logs");

// 遍历所有文件
foreach (string file in Directory.EnumerateFiles(@"C:\data", "*.txt", SearchOption.AllDirectories))
{
    Console.WriteLine(file);
}

// 删除目录(递归删除)
Directory.Delete(@"C:\data\logs", recursive: true);

‌2. 安全路径处理‌

// 合并路径(自动处理平台分隔符)
string path = Path.Combine("folder", "sub", "file.txt"); // 输出:folder/sub/file.txt

// 获取文件名或扩展名
string fileName = Path.GetFileName(path);    // file.txt
string ext = Path.GetExtension(path);        // .txt

‌六、异常处理与调试‌

‌1. 常见异常类型‌

异常类型触发场景
FileNotFoundException文件不存在
DirectoryNotFoundException目录不存在
UnauthorizedAccessException无权限访问文件或目录
IOException文件被占用、磁盘已满或路径过长

‌2. 健壮的异常处理‌

try
{
    string content = File.ReadAllText("config.json");
}
catch (FileNotFoundException ex)
{
    Console.WriteLine($"文件未找到: {ex.Message}");
}
catch (IOException ex)
{
    Console.WriteLine($"IO错误: {ex.Message}");
}
finally
{
    // 清理资源(如关闭数据库连接)
}

‌七、性能优化与最佳实践‌

‌1. 大文件处理‌

  • 使用流式读写‌:避免一次性加载大文件到内存。
  • ‌设置缓冲区大小‌:通过 FileStream 的 bufferSize 参数优化读写速度。

‌2. 安全性‌

  • ‌验证用户输入路径‌:防止路径遍历攻击(如 …/…/…/)。
string userInput = "userfile.txt";
string safePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, userInput);
  • ‌限制文件访问权限‌:使用 File.GetAccessControl 管理 ACL。

‌3. 资源释放‌

  • ‌始终使用 using 语句‌:确保流、读写器等资源被及时释放。
using (var stream = new FileStream("file.txt", FileMode.Open))
{
    // 操作流
} // 自动调用 stream.Dispose()

‌八、高级主题‌

‌1. 内存映射文件(Memory-Mapped Files)‌
高效处理超大文件或进程间共享数据:

using (var mmf = MemoryMappedFile.CreateFromFile("hugefile.bin"))
{
    using (var accessor = mmf.CreateViewAccessor())
    {
        // 直接操作内存映射区域
        accessor.Write(0, 123); // 写入整数
        int value = accessor.ReadInt32(0);
    }
}

‌2. 文件监控(FileSystemWatcher)‌
实时监控文件或目录变化:

var watcher = new FileSystemWatcher(@"C:\data");
watcher.Filter = "*.txt";
watcher.EnableRaisingEvents = true;

watcher.Changed += (sender, e) => 
{
    Console.WriteLine($"文件 {e.Name} 被修改!");
};

‌3. 文件压缩与解压‌
使用 System.IO.Compression 处理 ZIP 文件:

// 创建 ZIP 文件
ZipFile.CreateFromDirectory(@"C:\data", @"C:\archive.zip");

// 解压 ZIP 文件
ZipFile.ExtractToDirectory(@"C:\archive.zip", @"C:\extracted");

‌九、完整示例:文件加密与解密‌

using System.Security.Cryptography;
using System.Text;

public class FileEncryptor
{
    public static void EncryptFile(string inputFile, string outputFile, string password)
    {
        using (var aes = Aes.Create())
        {
            byte[] key = SHA256.Create().ComputeHash(Encoding.UTF8.GetBytes(password));
            byte[] iv = aes.IV;

            // 写入 IV 到输出文件
            using (var outputStream = new FileStream(outputFile, FileMode.Create))
            {
                outputStream.Write(iv, 0, iv.Length);

                // 加密数据
                using (var cryptoStream = new CryptoStream(outputStream, aes.CreateEncryptor(key, iv), CryptoStreamMode.Write))
                using (var inputStream = File.OpenRead(inputFile))
                {
                    inputStream.CopyTo(cryptoStream);
                }
            }
        }
    }

    public static void DecryptFile(string inputFile, string outputFile, string password)
    {
        using (var aes = Aes.Create())
        {
            byte[] key = SHA256.Create().ComputeHash(Encoding.UTF8.GetBytes(password));

            // 读取 IV
            using (var inputStream = new FileStream(inputFile, FileMode.Open))
            {
                byte[] iv = new byte[aes.IV.Length];
                inputStream.Read(iv, 0, iv.Length);

                // 解密数据
                using (var cryptoStream = new CryptoStream(inputStream, aes.CreateDecryptor(key, iv), CryptoStreamMode.Read))
                using (var outputStream = new FileStream(outputFile, FileMode.Create))
                {
                    cryptoStream.CopyTo(outputStream);
                }
            }
        }
    }
}

十、总结‌

1‌、核心原则‌

  • 资源管理‌:使用 using 或 try/finally 确保资源释放。
  • ‌异常处理‌:捕获特定异常,避免程序崩溃。
  • ‌性能优化‌:流式处理大文件,异步避免阻塞。

2‌、适用场景‌

  • ‌简单操作‌:优先使用 File 的静态方法。
  • ‌复杂控制‌:使用 FileStream 或 MemoryMappedFile。
  • ‌跨平台‌:通过 Path.Combine 处理路径分隔符。

掌握这些内容后,可以高效处理各类文件 I/O 需求,并规避常见陷阱。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2319175.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot之如何集成SpringDoc最详细文档

文章目录 一、概念解释1、OpenAPI2、Swagger3、Springfox4、Springdoc5. 关系与区别 二、SpringDoc基本使用1、导包2、正常编写代码&#xff0c;不需要任何注解3、运行后访问下面的链接即可 三、SpringDoc进阶使用1、配置文档信息2、配置文档分组3、springdoc的配置参数**1. 基…

【智能体】| 知识库、RAG概念区分以及智能体是什么

文章目录 前言简介大模型“幻觉”问题如何解决“幻觉”问题&#xff1f; RAG、智能体、RAG智能体概念什么是检索增强型生成&#xff08;RAG&#xff09;模拟简单的RAG场景 AI系统中的智能体是什么什么是Agentic RAG&#xff1f;Agentic RAG如何工作&#xff1f;Agentic RAG架构…

二分查找的应用

什么时候用二分查找&#xff1f; 数据具有二段性的时候 第一题&#xff1a; 题解代码&#xff1a; class Solution { public:int search(vector<int>& nums, int target) {int left 0,right nums.size()-1;while(left<right){int mid left (right-left)/2;//中…

【Function】Azure Function通过托管身份或访问令牌连接Azure SQL数据库

【Function】Azure Function通过托管身份或访问令牌连接Azure SQL数据库 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 【Function】Azure Function通过托管身份或访问令牌连接Azu…

Flink 通过 Chunjun Oracle LogMiner 实时读取 Oracle 变更日志并写入 Doris 的方案

文章目录 一、 技术背景二、 关键技术1、 Oracle LogMiner2、 Chunjun 的 LogMiner 关键流程3、修复 Chunjun Oracle LogMiner 问题 一、 技术背景 在大数据实时同步场景中&#xff0c;需要将 Oracle 数据库的变更数据&#xff08;CDC&#xff09; 采集并写入 Apache Doris&am…

WordPress系统获取webshell的攻略

一.后台修改模板拿WebShell 1.进入Vulhub靶场并执⾏以下命令开启靶场&#xff1b;在浏览器中访问并安装好 #执⾏命令 cd /vulhub/wordpress/pwnscriptum docker-compose up -d 2. 修改其WP的模板&#xff0c;登陆WP后点击 【外 观】 --》 【编辑】 --》 404.php 3.插入一句话木…

蓝桥杯2023年第十四届省赛真题-子矩阵

题目来自DOTCPP&#xff1a; 暴力思路&#xff08;两个测试点超时&#xff09;&#xff1a; 题目要求我们求出子矩阵的最大值和最小值的乘积&#xff0c;我们可以枚举矩阵中的所有点&#xff0c;以这个点为其子矩阵的左上顶点&#xff0c;然后判断一下能不能构成子矩阵。如果可…

如何在 Node.js 中使用 .env 文件管理环境变量 ?

Node.js 应用程序通常依赖于环境变量来管理敏感信息或配置设置。.env 文件已经成为一种流行的本地管理这些变量的方法&#xff0c;而无需在代码存储库中公开它们。本文将探讨 .env 文件为什么重要&#xff0c;以及如何在 Node.js 应用程序中有效的使用它。 为什么使用 .env 文…

Redis BitMap 用户签到

Redis Bitmap Bitmap&#xff08;位图&#xff09;是 Redis 提供的一种用于处理二进制位&#xff08;bit&#xff09;的特殊数据结构&#xff0c;它基于 String 类型&#xff0c;每个 bit 代表一个布尔值&#xff08;0 或 1&#xff09;&#xff0c;可以用于存储大规模的二值状…

未来办公与生活的新范式——智慧园区

在信息化与智能化技术飞速发展的今天&#xff0c;智慧园区作为一种新兴的城市发展形态&#xff0c;正逐步成为推动产业升级、提升城市管理效率、改善居民生活质量的重要力量。智慧园区不仅融合了先进的信息技术&#xff0c;还深刻体现了可持续发展的理念&#xff0c;为园区内的…

Hugging Face预训练GPT微调ChatGPT(微调入门!新手友好!)

Hugging Face预训练GPT微调ChatGPT&#xff08;微调入门&#xff01;新手友好&#xff01;&#xff09; 在实战中&#xff0c;⼤多数情况下都不需要从0开始训练模型&#xff0c;⽽是使⽤“⼤⼚”或者其他研究者开源的已经训练好的⼤模型。 在各种⼤模型开源库中&#xff0c;最…

【CSS3】化神篇

目录 平面转换平移旋转改变旋转原点多重转换缩放倾斜 渐变线性渐变径向渐变 空间转换平移视距旋转立体呈现缩放 动画使现步骤animation 复合属性animation 属性拆分逐帧动画多组动画 平面转换 作用&#xff1a;为元素添加动态效果&#xff0c;一般与过渡配合使用 概念&#x…

Unity音频混合器如何暴露参数

音频混合器是Unity推荐管理音效混音的工具&#xff0c;那么如何使用代码对它进行管理呢&#xff1f; 首先我在AudioMixer的Master组中创建了BGM和SFX的分组&#xff0c;你也可以直接用Master没有问题。 这里我以BGM为例&#xff0c;如果要在代码中进行使用就需要将参数暴露出去…

如何理解分布式光纤传感器?

关键词&#xff1a;OFDR、分布式光纤传感、光纤传感器 分布式光纤传感器是近年来备受关注的前沿技术&#xff0c;其核心在于将光纤本身作为传感介质和信号传输介质&#xff0c;通过解析光信号在光纤中的散射效应&#xff0c;实现对温度、应变、振动等物理量的连续、无盲区、高…

PMP-项目运行环境

你好&#xff01;我是 Lydia-穎穎 ♥感谢你的陪伴与支持 ~~~ 欢迎一起探索未知的知识和未来&#xff0c;现在lets go go go!!! 1. 影响项目的要素 项目存在在不同的环境下&#xff0c;环境对于项目的交付产生不同的影响。需了解环境对于项目的影响&#xff0c;采取相应措施应对…

shell 脚本搭建apache

#!/bin/bash # Set Apache version to install ## author: yuan# 检查外网连接 echo "检查外网连接..." ping www.baidu.com -c 3 > /dev/null 2>&1 if [ $? -eq 0 ]; thenecho "外网通讯良好&#xff01;" elseecho "网络连接失败&#x…

Huawei 鲲鹏(ARM/Aarch64)服务器安装KVM虚拟机(非桌面视图)

提出问题 因需要进行ARM架构适配&#xff0c;需要在Huawei Taishan 200k&#xff08;CPU&#xff1a; Kunpeng 920 5231K&#xff09;上&#xff0c;创建几台虚拟机做为开发测试环境。 无奈好久没搞了&#xff0c;看了一下自己多年前写的文章&#xff1a;Huawei 鲲鹏&#xf…

《Python实战进阶》No28: 使用 Paramiko 实现远程服务器管理

No28: 使用 Paramiko 实现远程服务器管理 摘要 在现代开发与运维中&#xff0c;远程服务器管理是必不可少的一环。通过 SSH 协议&#xff0c;我们可以安全地连接到远程服务器并执行各种操作。Python 的 Paramiko 模块是一个强大的工具&#xff0c;能够帮助我们实现自动化任务&…

【Kafka】深入了解Kafka

集群的成员关系 Kafka使用Zookeeper维护集群的成员信息。 每一个broker都有一个唯一的标识&#xff0c;这个标识可以在配置文件中指定&#xff0c;也可以自动生成。当broker在启动时通过创建Zookeeper的临时节点把自己的ID注册到Zookeeper中。broker、控制器和其他一些动态系…

C++特性——RAII、智能指针

RAII 就像new一个需要delete&#xff0c;fopen之后需要fclose&#xff0c;但这样会有隐形问题&#xff08;忘记释放&#xff09;。RAII即用对象把这个过程给包起来&#xff0c;对象构造的时候&#xff0c;new或者fopen&#xff0c;析构的时候delete. 为什么需要智能指针 对于…