C #: переход по строкам многострочной строки


100

Каков хороший способ перебрать каждую строку многострочной строки без использования гораздо большего объема памяти (например, без разделения ее на массив)?

Ответы:


159

Я предлагаю использовать комбинацию StringReaderи мой LineReaderкласс, который является частью MiscUtil, но также доступен в этом ответе StackOverflow - вы можете легко скопировать только этот класс в свой собственный служебный проект. Вы бы использовали это так:

string text = @"First line
second line
third line";

foreach (string line in new LineReader(() => new StringReader(text)))
{
    Console.WriteLine(line);
}

Цикл по всем строкам в теле строковых данных (будь то файл или что-то еще) настолько распространен, что не должен требовать, чтобы вызывающий код проверял на ноль и т. Д. :) Сказав это, если вы действительно хотите сделать ручной цикл, это форма, которую я обычно предпочитаю Фредрику:

using (StringReader reader = new StringReader(input))
{
    string line;
    while ((line = reader.ReadLine()) != null)
    {
        // Do something with the line
    }
}

Таким образом, вам нужно только один раз проверить значение null, и вам не нужно думать о цикле do / while (который по какой-то причине всегда требует больше усилий для чтения, чем простой цикл while).


74

Вы можете использовать a, StringReaderчтобы читать строку за раз:

using (StringReader reader = new StringReader(input))
{
    string line = string.Empty;
    do
    {
        line = reader.ReadLine();
        if (line != null)
        {
            // do something with the line
        }

    } while (line != null);
}

1
Большой; +1; это помогло; но я просто хочу добавить, что на самом деле нет необходимости использовать блок «using», потому что в этом случае нет никаких ресурсов, которые можно было бы закрыть. См. Примечания в статье StringReader на docs.microsoft.com
RD Alkire

10

Я знаю, что на это есть ответ, но я хотел бы добавить свой ответ:

using (var reader = new StringReader(multiLineString))
{
    for (string line = reader.ReadLine(); line != null; line = reader.ReadLine())
    {
        // Do something with the line
    }
}

7

из MSDN для StringReader

    string textReaderText = "TextReader is the abstract base " +
        "class of StreamReader and StringReader, which read " +
        "characters from streams and strings, respectively.\n\n" +

        "Create an instance of TextReader to open a text file " +
        "for reading a specified range of characters, or to " +
        "create a reader based on an existing stream.\n\n" +

        "You can also use an instance of TextReader to read " +
        "text from a custom backing store using the same " +
        "APIs you would use for a string or a stream.\n\n";

    Console.WriteLine("Original text:\n\n{0}", textReaderText);

    // From textReaderText, create a continuous paragraph 
    // with two spaces between each sentence.
    string aLine, aParagraph = null;
    StringReader strReader = new StringReader(textReaderText);
    while(true)
    {
        aLine = strReader.ReadLine();
        if(aLine != null)
        {
            aParagraph = aParagraph + aLine + " ";
        }
        else
        {
            aParagraph = aParagraph + "\n";
            break;
        }
    }
    Console.WriteLine("Modified text:\n\n{0}", aParagraph);

2

Вот небольшой фрагмент кода, который найдет первую непустую строку в строке:

string line1;
while (
    ((line1 = sr.ReadLine()) != null) &&
    ((line1 = line1.Trim()).Length == 0)
)
{ /* Do nothing - just trying to find first non-empty line*/ }

if(line1 == null){ /* Error - no non-empty lines in string */ }

2

Чтобы обновить этот древний вопрос для .NET 4, теперь есть более простой способ:

var lines = File.ReadAllLines(filename);

foreach (string line in lines)
{
    Console.WriteLine(line);
}

0

Попробуйте использовать метод String.Split:

string text = @"First line
second line
third line";

foreach (string line in text.Split('\n'))
{
    // do something
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.