Jak znaleźć i zamienić tekst w pliku za pomocą C #


157

Jak dotąd mój kod

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

Wiem, jak znaleźć tekst, ale nie mam pojęcia, jak zamienić tekst w pliku na własny.


Rozważmy następujący komentarz jako jedyna wskazówka: Jeśli masz visual studio, można dołączyć foldery w roztworze i skorzystać z wyszukiwania i zamiany funkcji z visual studio .Best szczęścia
StackOrder

Odpowiedzi:


321

Przeczytaj całą zawartość pliku. Dokonaj wymiany za pomocą String.Replace. Zapisz zawartość z powrotem do pliku.

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

5
@WinCoder BTW dla bardziej złożonych zamienników, których możesz użyćRegex.Replace
Sergey Berezovskiy

35
Powoduje to jednoczesny odczyt całego pliku do pamięci, nie zawsze tak dobrze.
Banshee

6
@Banshee Touche „Właśnie próbowałem odczytać 9 000 000 wierszy i został wyrzucony System out of memorywyjątek.
Squ1rr3lz

4
W przypadku dużych plików jest to bardziej złożony problem. Czytaj fragmenty bajtów, analizuj je, czytaj kolejny fragment, itp.
Alexander

6
@Alexander Right. Jeden fragment kończy się na „... som”, a następny zaczyna się na „e text…”. Sprawia, że ​​jest to znacznie bardziej skomplikowany problem.
djv

36

Będzie ci ciężko pisać do tego samego pliku, z którego czytasz. Jednym z szybkich sposobów jest zrobienie tego:

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

Możesz to lepiej rozłożyć

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

3
Jest to proste, ale niepożądane w przypadku bardzo dużych plików. (ps. nie jestem tym, który przegłosował)
Alvin Wong,

3
Zgadzam się, ale podczas czytania nie możesz pisać do pliku. O ile nie napiszesz do innego pliku, a następnie zastąpisz go zmianą nazwy ... tak czy inaczej, nowy plik musi być przechowywany w innym miejscu podczas tworzenia, czy to w pamięci, czy na dysku.
Flynn1179

@ Flynn1179 Nieprawda w tym przykładzie. To działa. Wypróbuj to. Myślę, że ReadAllTextzamyka dostęp do plików przed WriteAllText. Używam tej właśnie techniki we własnej aplikacji.
SteveCinq

Wiem; ten przykład nie pisze podczas czytania, o to mi chodziło!
Flynn1179

27

Musisz zapisać wszystkie czytane wiersze do pliku wyjściowego, nawet jeśli ich nie zmieniasz.

Coś jak:

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);
  }
}

Jeśli chcesz wykonać tę operację na miejscu, najłatwiejszym sposobem jest użycie tymczasowego pliku wyjściowego i na końcu zastąpienie pliku wejściowego wynikiem.

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

(Próba wykonania operacji aktualizacji w środku pliku tekstowego jest raczej trudna do wykonania, ponieważ zawsze posiadanie zamiennika tej samej długości jest trudne, biorąc pod uwagę, że większość kodowań ma zmienną szerokość).

EDYCJA: Zamiast dwóch operacji na plikach w celu zastąpienia oryginalnego pliku, lepiej użyć File.Replace("input.txt", "output.txt", null). (Zobacz MSDN ).


1
VB musiał zmienić 2 linie: Używanie wejścia jako nowego StreamReader (nazwa pliku) Podczas input.Peek ()> = 0
Brent

8

Prawdopodobnie będziesz musiał przeciągnąć plik tekstowy do pamięci, a następnie dokonać wymiany. Będziesz wtedy musiał nadpisać plik przy użyciu metody, o której dobrze wiesz. Więc najpierw:

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

Następnie możesz przejrzeć i zamienić tekst w tablicy.

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);
}

ta metoda daje ci pewną kontrolę nad możliwymi manipulacjami. Lub możesz po prostu dokonać zamiany w jednej linii

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

Mam nadzieję, że to pomoże.


6

Tak to zrobiłem z dużym (50 GB) plikiem:

Wypróbowałem 2 różne sposoby: pierwszy, wczytanie pliku do pamięci i użycie Regex Replace lub String Replace. Następnie dołączyłem cały ciąg do pliku tymczasowego.

Pierwsza metoda działa dobrze w przypadku kilku zamian Regex, ale Regex.Replace lub String.Replace może spowodować błąd braku pamięci, jeśli wykonasz wiele zamian w dużym pliku.

Drugi polega na odczytaniu pliku tymczasowego wiersz po wierszu i ręcznym utworzeniu każdego wiersza przy użyciu StringBuilder i dołączeniu każdego przetworzonego wiersza do pliku wynikowego. Ta metoda była dość szybka.

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

Ten kod zadziałał dla mnie

- //-------------------------------------------------------------------
                           // 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

staram się używać prostego kodu do przodu tak często, jak tylko mogę, poniższy kod działał dobrze ze mną

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();
}
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.