Как найти и заменить текст в файле с C #


157

Мой код пока

StreamReader reading = File.OpenText("test.txt");
string str;
while ((str = reading.ReadLine())!=null)
{
      if (str.Contains("some text"))
      {
          StreamWriter write = new StreamWriter("test.txt");
      }
}

Я знаю, как найти текст, но понятия не имею, как заменить текст в файле своим собственным.


Рассматривайте этот комментарий только как совет: если у вас есть Visual Studio, вы можете включить в решение папки и использовать функцию поиска и замены в Visual Studio. Удачи
StackOrder

Ответы:


322

Читать все содержимое файла. Сделайте замену с String.Replace. Записать содержимое обратно в файл.

string text = File.ReadAllText("test.txt");
text = text.Replace("some text", "new value");
File.WriteAllText("test.txt", text);

5
@WinCoder BTW для более сложных замен, которые вы можете использоватьRegex.Replace
Сергей Березовский

35
Это сразу читает весь файл в память, не всегда это хорошо.
Банши

6
@Banshee Touche 'Я только что попытался прочитать 9 000 000 строк, и меня бросили в System out of memoryисключение.
Squ1rr3lz

4
Для больших файлов это более сложная проблема. Чтение байтового фрагмента, анализ их, чтение другого фрагмента и т. Д.
Александр

6
@ Александр Верно. Один кусок заканчивается на «... сом», а следующий начинается на «е текст ...». Делает это гораздо более сложной проблемой.
DJV

36

Вам будет трудно писать в тот же файл, из которого вы читаете. Один быстрый способ - просто сделать это:

File.WriteAllText("test.txt", File.ReadAllText("test.txt").Replace("some text","some other text"));

Вы можете выложить это лучше с

string str = File.ReadAllText("test.txt");
str = str.Replace("some text","some other text");
File.WriteAllText("test.txt", str);

3
Это просто, но не желательно для очень больших файлов. (ps Я не тот, кто понизил)
Элвин Вонг

3
Я бы согласился, но вы не можете писать в файл, пока читаете из него. Если вы не записываете в другой файл, а затем заменяете его переименованием ... в любом случае, новый файл должен храниться где-то еще во время его создания, будь то в памяти или на диске.
Flynn1179

@ Flynn1179 Не верно в этом примере. Оно работает. Попробуйте это. Я предполагаю, что ReadAllTextзакрывает доступ к файлу до WriteAllText. Я использую эту технику в своем собственном приложении.
SteveCinq

Я знаю; этот пример не пишет, пока он читает, это была моя точка зрения!
Flynn1179

27

Вам нужно записать все строки, которые вы прочитали, в выходной файл, даже если вы их не меняете.

Что-то вроде:

using (var input = File.OpenText("input.txt"))
using (var output = new StreamWriter("output.txt")) {
  string line;
  while (null != (line = input.ReadLine())) {
     // optionally modify line.
     output.WriteLine(line);
  }
}

Если вы хотите выполнить эту операцию на месте, то самый простой способ - использовать временный выходной файл и в конце заменить входной файл выходным.

File.Delete("input.txt");
File.Move("output.txt", "input.txt");

(Попытка выполнить операции обновления в середине текстового файла довольно трудна для понимания, потому что всегда иметь замену одинаковой длины сложно, поскольку большинство кодировок имеют переменную ширину.)

РЕДАКТИРОВАТЬ: Вместо двух файловых операций, чтобы заменить исходный файл, лучше использовать File.Replace("input.txt", "output.txt", null). (См. MSDN .)


1
VB пришлось изменить 2 строки: использование ввода как нового StreamReader (имя файла), а input.Peek ()> = 0
Brent

8

Вероятно, вам придется вытянуть текстовый файл в память, а затем сделать замены. Затем вам придется перезаписать файл, используя метод, который вам известен. Итак, вы бы сначала:

// Read lines from source file.
string[] arr = File.ReadAllLines(file);

Затем вы можете перебрать и заменить текст в массиве.

var writer = new StreamWriter(GetFileName(baseFolder, prefix, num));
for (int i = 0; i < arr.Length; i++)
{
    string line = arr[i];
    line.Replace("match", "new value");
    writer.WriteLine(line);
}

Этот метод дает вам некоторый контроль над манипуляциями, которые вы можете сделать. Или вы можете просто сделать замену в одной строке

File.WriteAllText("test.txt", text.Replace("match", "new value"));

Надеюсь, это поможет.


6

Вот как я это сделал с большим (50 ГБ) файлом:

Я пробовал 2 разных способа: первый - чтение файла в память и использование Regex Replace или String Replace. Затем я добавил всю строку во временный файл.

Первый метод работает хорошо для нескольких замен Regex, но Regex.Replace или String.Replace могут вызвать ошибку нехватки памяти, если вы делаете много замен в большом файле.

Во-вторых, построчно читая временный файл и вручную строя каждую строку, используя StringBuilder, и добавляя каждую обработанную строку в файл результатов. Этот метод был довольно быстрым.

static void ProcessLargeFile()
{
        if (File.Exists(outFileName)) File.Delete(outFileName);

        string text = File.ReadAllText(inputFileName, Encoding.UTF8);

        // EX 1 This opens entire file in memory and uses Replace and Regex Replace --> might cause out of memory error

        text = text.Replace("</text>", "");

        text = Regex.Replace(text, @"\<ref.*?\</ref\>", "");

        File.WriteAllText(outFileName, text);




        // EX 2 This reads file line by line 

        if (File.Exists(outFileName)) File.Delete(outFileName);

        using (var sw = new StreamWriter(outFileName))      
        using (var fs = File.OpenRead(inFileName))
        using (var sr = new StreamReader(fs, Encoding.UTF8)) //use UTF8 encoding or whatever encoding your file uses
        {
            string line, newLine;

            while ((line = sr.ReadLine()) != null)
            {
              //note: call your own replace function or use String.Replace here 
              newLine = Util.ReplaceDoubleBrackets(line);

              sw.WriteLine(newLine);
            }
        }
    }

    public static string ReplaceDoubleBrackets(string str)
    {
        //note: this replaces the first occurrence of a word delimited by [[ ]]

        //replace [[ with your own delimiter
        if (str.IndexOf("[[") < 0)
            return str;

        StringBuilder sb = new StringBuilder();

        //this part gets the string to replace, put this in a loop if more than one occurrence  per line.
        int posStart = str.IndexOf("[[");
        int posEnd = str.IndexOf("]]");
        int length = posEnd - posStart;


        // ... code to replace with newstr


        sb.Append(newstr);

        return sb.ToString();
    }

0

Этот код работал для меня

- //-------------------------------------------------------------------
                           // Create an instance of the Printer
                           IPrinter printer = new Printer();

                           //----------------------------------------------------------------------------
                           String path = @"" + file_browse_path.Text;
                         //  using (StreamReader sr = File.OpenText(path))

                           using (StreamReader sr = new System.IO.StreamReader(path))
                           {

                              string fileLocMove="";
                              string newpath = Path.GetDirectoryName(path);
                               fileLocMove = newpath + "\\" + "new.prn";



                                  string text = File.ReadAllText(path);
                                  text= text.Replace("<REF>", reference_code.Text);
                                  text=   text.Replace("<ORANGE>", orange_name.Text);
                                  text=   text.Replace("<SIZE>", size_name.Text);
                                  text=   text.Replace("<INVOICE>", invoiceName.Text);
                                  text=   text.Replace("<BINQTY>", binQty.Text);
                                  text = text.Replace("<DATED>", dateName.Text);

                                       File.WriteAllText(fileLocMove, text);



                               // Print the file
                               printer.PrintRawFile("Godex G500", fileLocMove, "n");
                              // File.WriteAllText("C:\\Users\\Gunjan\\Desktop\\new.prn", s);
                           }

0

Я стараюсь использовать простой прямой код столько, сколько я могу, ниже код работал нормально со мной

using System;
using System.IO;
using System.Text.RegularExpressions;

/// <summary>
/// Replaces text in a file.
/// </summary>
/// <param name="filePath">Path of the text file.</param>
/// <param name="searchText">Text to search for.</param>
/// <param name="replaceText">Text to replace the search text.</param>
static public void ReplaceInFile( string filePath, string searchText, string replaceText )
{
    StreamReader reader = new StreamReader( filePath );
    string content = reader.ReadToEnd();
    reader.Close();

    content = Regex.Replace( content, searchText, replaceText );

    StreamWriter writer = new StreamWriter( filePath );
    writer.Write( content );
    writer.Close();
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.