Mono BLOG

Blog about Microsoft technologies (.NET, ASP.NET Core, Blazor, EF Core, WPF, TypeScript, etc.)

C# String Manipulation

in .NET

Working with strings is a daily task for developers, but mastering efficient and error-free string operations in C# requires understanding key techniques and avoiding common mistakes. This guide explores best practices for optimizing string handling.


1. Prefer char Over string for Single Characters

Strings are reference types (heap-allocated), while char is a value type (stack-allocated). Use char-based overloads for better performance:

string str = "Hello, World!";
bool startsWithH = str.StartsWith('H');  // Faster than StartsWith("H")
bool containsO = str.Contains('o');     // More efficient than Contains("o")
int index = str.IndexOf('o');

Why? • Avoids unnecessary heap allocations. • Direct value comparisons skip substring checks.


2. Leverage Method Overloads to Reduce Overhead

Many string methods offer optimized overloads. Avoid redundant operations:

Example 1: Splitting Strings

string input = "  Hello, World,, Good, Morning ";

// ❌ Inefficient: Filters empty entries post-split
var slices = input.Split(',').Where(s => !string.IsNullOrEmpty(s)).ToArray();

// ✅ Optimized: Uses built-in options
var optimizedSlices = input.Split(',', StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);

Example 2: Case-Insensitive Comparisons

// ❌ Creates temporary strings
if (s1.ToLower() == s2.ToLower()) { ... }

// ✅ Direct comparison with no allocations
if (s1.Equals(s2, StringComparison.OrdinalIgnoreCase)) { ... }

3. Use string Constructors Wisely

Create strings efficiently using constructors for repeated characters or arrays:

// Create "==========" (20 '=' characters)
string separator = new string('=', 20);  

// Reverse a string
char[] chars = originalStr.ToCharArray();
Array.Reverse(chars);
string reversed = new string(chars);  // More efficient than string.Join("", str.Reverse())

4. Handle OS-Specific Scenarios

Use platform-agnostic APIs for paths and line breaks:

Paths

string path = Path.Combine("folder", "subfolder/", "file.txt"); 
// Automatically resolves to "folder/subfolder/file.txt" (Unix) or "folder\subfolder\file.txt" (Windows)

Line Endings

string text = "Line1\r\nLine2\nLine3";
string normalized = text.ReplaceLineEndings();  // Converts to Environment.NewLine

5. Optimize StringBuilder Usage

Avoid intermediate allocations with StringBuilder’s advanced features:

var sb = new StringBuilder();
sb.Append("Hello")
  .Append('!', 3)         // Appends "!!!"
  .Insert(0, "[START] ")  // Result: "[START] Hello!!!"
  .Replace("START", "END");

string result = sb.ToString();

Key Methods: • AppendFormat(): For formatted strings. • AppendJoin(): Combine collections with separators.


6. Embrace String Interpolation

Prioritize readability and performance with interpolated strings:

int id = 123;
string name = "Alice";
string details = $"ID: {id}, Name: {name}";  // Compiler optimizes this

Performance Note: Interpolation outperforms string.Format() and is cleaner than StringBuilder for simple cases.


Common Pitfalls to Avoid

MistakeSolution
Using + for large concatenationsUse StringBuilder
Ignoring culture in comparisonsSpecify StringComparison (e.g., Ordinal or CurrentCulture)
Hardcoding path separatorsUse Path.Combine() and Path.DirectorySeparatorChar
Unnecessary substring allocationsUse AsSpan() and Range for read-only operations

Summary

Performance: Prefer char over string, use method overloads, and leverage StringBuilder. • Cross-Platform: Rely on Path and Environment classes for OS-specific behaviors. • Readability: Use interpolated strings for clarity without sacrificing efficiency.

By applying these techniques, you’ll write more efficient, maintainable, and platform-resilient C# code. For advanced scenarios, explore Span<char>, StringPool, and encoding APIs like System.Text.Encoding.