C# 使用 Filestream 快速修改大文件中某一行指定位置的数据

4 篇文章 0 订阅

前言

C# 中读写文件常规流程

FileStream fs = new FileStream(filePath, FileMode.OpenOrCreate, FileAccess.ReadWrite);
//读
StreamReader sr=new StreamReader(fs);
string line=sr.readLine();

//写
StreamWriter s = new StreamWriter(fs);
s.WriteLine("xxx");

至于修改内容 网上大多数是先把文件所有内容读进内存,再把内容修改后写入文件,这样对于小文件没有问题,但是对于大文件就不可行了;

 

操作步骤

  1. 我们首先写一个函数来读取一行数据
   /// <summary>
   /// 在文件流当前位置开始读取改行的内容 返回改行的长度
   /// </summary>  
   private long readRow(FileStream fs20O)
   {
            long len = 0;

            byte[] b = new byte[1];

            bool exit = false;

            while (true)
            {
                if (fs20O.Read(b, 0, 1) == -1) break;

                len++;
                string str = Encoding.UTF8.GetString(b);
                Console.Write(str);
                if ("\n".Equals(str))
                {
                    return len;
                }
            }

            return len;
     }

然后有人会问了,为啥我不用 StreamReader.readLine()来读取一行呢?

由于接下来需要修改当前行的内容,读取完当前行并确定需要修改的时候,需要把流的位置移动至行首,根据上面函数得到读取的长度,使用代码

 fs.Seek(-len, SeekOrigin.Current);

如果用len=StreamReader.readLine(),再移动指针,你会发现流的位置会在更后面而不是偏移了len,不知道readLine是不是会预读取,导致你觉得他只读取了一行,实际他把流的指针移到后几行了并且缓存了数据,再做偏移的时候 结果就不对了 这就是为什么要用上面的方法读取了;

2. 好了,接下来遍历读取文件内容

long len=0;
while((len=readRow(fs)!=-1)
{
    //......
    //修改内容
    //文件指针移动至行首
    if(true)
    {
        fs.Seek(-len, SeekOrigin.Current);
        byte[] readWriteByte = Encoding.UTF8.GetBytes("Test");
        //会直接把原来内容替换
        fs.Write(readWriteByte, 0, readWriteByte.Length);
        //***继续往下读取 直到行尾 把指针移到下一行行首
        readRow(fs);

        //修改完成直接跳出循环,保存文件
        break;
    }
}
  • 2
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用FileStream读取CSV文件,需要使用StreamReader将FileStream转化为文本流,然后逐读取CSV文件内容。找到并清空带有指定字符串的那一行数据,需要先找到该数据所在的数,然后再将该数据清空。 以下是一个示例代码,演示如何使用FileStream和StreamReader读取CSV文件,并清空指定字符串所在的数据: ```csharp using (FileStream fs = new FileStream("data.csv", FileMode.OpenOrCreate, FileAccess.ReadWrite)) { using (StreamReader sr = new StreamReader(fs)) { // 读取CSV文件内容 string line; int lineNumber = 0; while ((line = sr.ReadLine()) != null) { lineNumber++; // 判断是否包含指定字符串 if (line.Contains("指定字符串")) { // 清空该数据 fs.Seek(-(line.Length + 2), SeekOrigin.Current); // 退回到该数据的起始位置 for (int i = 0; i < line.Length + 2; i++) // 将该数据清空 { fs.WriteByte(0); } break; // 找到并清空数据后退出循环 } } } } ``` 在上面的代码,我们使用了`FileStream.Seek`方法将文件指针退回到该数据的起始位置,然后使用`FileStream.WriteByte`方法将该数据清空。由于CSV文件是以逗号分隔的,所以我们需要将每数据的结尾多加两个字节,即`"\r\n"`,所以在清空该数据时,需要将该数据长度加上2。 注意:在清空该数据时,如果该数据的长度不足以填满整,可能会导致数据错位。因此,建议在写入CSV文件时,确保每数据的长度相等,或者在数据结尾处补足空格。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值