Let's say I have two strings: a and b. To compare whether a and be have the same values when case is ignored, I've always used:
// (Assume a and b have been verified not to be null)
if (a.ToLower() == b.ToLower())
However, using Reflector, I've seen this a few times in the .NET Framework:
// (arg three is ignoreCase)
if (string.Compare(a, b, true) == 0)
I tested which is faster, and the ToLower() beat Compare() every time with the strings I used.
Is there a reason why to Compare() instead of ToLower()? Something about different CultureInfo? I'm scratching my head.
6 Answers 6
The main thing you should be concerned about isn't performance, it's correctness, and from that aspect the method you probably want to be using for a case insensitive comparison is either:
string.Compare(a, b, StringComparison.OrdinalIgnoreCase) == 0;
or
a.Equals(b, StringComparison.OrdinalIgnoreCase)
(The first one is useful if you know the strings may be null; the latter is simpler to write if you already know that at least one string is non-null. I've never tested the performance but assume it will be similar.)
Ordinal or OrdinalIgnoreCase are a safe bet unless you know you want to use another comparison method; to get the information needed to make the decision read this article on MSDN.
3 Comments
NullReferenceException instead of an incorrect result. Because you cannot compare nothing with something. In the odd cases where null makes sense I'd check for null beforehand and return false rather than replacing it with a compare function that is used to replace Equals, obfuscating the code.The Remarks section of the MSDN article should explain things. Essentially, the reason is for compatibility across different cultures settings.
Comments
When comparing strings you should always use an explicit StringComparison member. The String functions are somewhat inconsistent in how they choose to compare strings. The only way to guarantee the comparision used is to a) memorize all of them (this includes both you and everyone on your team) or b) use an explicit comparison for every function.
It's much better to be explicit and not rely on group knowledge being perfect. Your teammates will thank you for this.
Example:
if ( StringComparison.OrdinalIgnoreCase.Equals(a,b) )
Using ToLower for comparison has 2 problems I can think of off the top of my head
- It allocates memory. Comparison functions should not allocate memory unless they absolutely have to.
- Strings can be lowered in several ways. Most notable Ordinal or Culture Sensitive lower. Which way does .ToLower() work? Personally, I don't know. It's much better to pass an explicit culture than rely on the default.
2 Comments
String.Equals(a, b, StringComparison.OrdinalIgnoreCase.Equals)?OrdinalIgnoreCase actually does either. I can hazard a guess that it only ups or lowers ASCII or Latin, but it would remain a guess.Another MSDN article that provides some DOs and DON'Ts and recommendations for what comparison method to use in various cases: New Recommendations for Using Strings in Microsoft .NET 2.0
Comments
ToLower() is not a comparison function, it puts the string to lower case. When the == operator is used on String objects in C# it is optimized by the compiler. At the core, Both depend on System.String.Equals as seen in Reflector.
Could your post your test that shows calling ToLower() is faster than a case-insensitive comparison? My tests show the opposite to be true! Regardless, other posters' points about correctness stand.