What is a good way to loop through each line of a multiline string without using much more memory (for example without splitting it into an array)?
8 Answers
I suggest using a combination of StringReader
and my LineReader
class, which is part of MiscUtil but also available in this StackOverflow answer - you can easily copy just that class into your own utility project. You'd use it like this:
string text = @"First line
second line
third line";
foreach (string line in new LineReader(() => new StringReader(text)))
{
Console.WriteLine(line);
}
Looping over all the lines in a body of string data (whether that's a file or whatever) is so common that it shouldn't require the calling code to be testing for null etc :) Having said that, if you do want to do a manual loop, this is the form that I typically prefer over Fredrik's:
using (StringReader reader = new StringReader(input))
{
string line;
while ((line = reader.ReadLine()) != null)
{
// Do something with the line
}
}
This way you only have to test for nullity once, and you don't have to think about a do/while loop either (which for some reason always takes me more effort to read than a straight while loop).
-
Looks like MiscUtil doesn't have a version for .NET Core. – void.pointer Mar 13 '21 at 20:14
-
@void.pointer: No, it doesn't I'm afraid. It's not a project I'been putting any time into over the last 10 years. – Jon Skeet Mar 14 '21 at 07:31
-
StringReader, an object little be know it. Thanks. That was What I need. – RogerEdward May 11 '21 at 07:18
You can use a StringReader
to read a line at a time:
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);
}

- 155,851
- 29
- 291
- 343
-
3Great; +1; this helped; but I just want to add that one does not actually need to use the "using" block because there aren't any resources to close in this case. See [remarks in StringReader article at learn.microsoft.com](https://learn.microsoft.com/en-us/dotnet/api/system.io.stringreader?view=netcore-3.1#remarks) – R.D. Alkire Jan 29 '20 at 21:50
I know this has been answered, but I'd like to add my own answer:
using (var reader = new StringReader(multiLineString))
{
for (string line = reader.ReadLine(); line != null; line = reader.ReadLine())
{
// Do something with the line
}
}

- 351
- 2
- 4
from MSDN for 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);

- 17,883
- 5
- 53
- 72
Here's a quick code snippet that will find the first non-empty line in a string:
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 */ }

- 11,856
- 6
- 53
- 77
Try using String.Split Method:
string text = @"First line
second line
third line";
foreach (string line in text.Split('\n'))
{
// do something
}

- 21
- 2
Sometimes I think we can overcomplicate the solution just to avoid repeating one line of code. This is the reason I landed on this question in the first place.
After thinking about it for a bit I came to the conclusion that the simplest solution is to repeat the ReadLine
before and inside the loop.
using (var stringReader = new StringReader(input))
{
var line = await stringReader.ReadLineAsync();
while (line != null)
{
// do something
line = await stringReader.ReadLineAsync();
}
}
I realize this might be considered to not follow the DRY principle, but I think it's worth considering given the simplicity.

- 9,437
- 4
- 41
- 52
You could use an extension method:
public static IEnumerable<string> EnumerateLines(this string source)
{
ArgumentNullException.ThrowIfNull(source);
using StringReader reader = new StringReader(source);
while(reader.ReadLine() is { } line)
{
yield return line;
}
}

- 1