2 // System.StringComparer
5 // Marek Safar (marek.safar@seznam.cz)
7 // (C) 2005 Novell, Inc (http://www.novell.com)
10 // Permission is hereby granted, free of charge, to any person obtaining
11 // a copy of this software and associated documentation files (the
12 // "Software"), to deal in the Software without restriction, including
13 // without limitation the rights to use, copy, modify, merge, publish,
14 // distribute, sublicense, and/or sell copies of the Software, and to
15 // permit persons to whom the Software is furnished to do so, subject to
16 // the following conditions:
18 // The above copyright notice and this permission notice shall be
19 // included in all copies or substantial portions of the Software.
21 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
32 using System.Collections;
33 using System.Globalization;
34 using System.Runtime.InteropServices;
35 using System.Collections.Generic;
39 [Serializable, ComVisible(true)]
40 public abstract class StringComparer : IComparer, IEqualityComparer, IComparer<string>, IEqualityComparer<string>
42 static StringComparer invariantCultureIgnoreCase = new CultureAwareComparer (CultureInfo.InvariantCulture, true);
43 static StringComparer invariantCulture = new CultureAwareComparer (CultureInfo.InvariantCulture, false);
44 static StringComparer ordinalIgnoreCase = new OrdinalComparer (true);
45 static StringComparer ordinal = new OrdinalComparer (false);
48 protected StringComparer ()
53 public static StringComparer CurrentCulture {
55 return new CultureAwareComparer (CultureInfo.CurrentCulture, false);
59 public static StringComparer CurrentCultureIgnoreCase {
61 return new CultureAwareComparer (CultureInfo.CurrentCulture, true);
65 public static StringComparer InvariantCulture {
67 return invariantCulture;
71 public static StringComparer InvariantCultureIgnoreCase {
73 return invariantCultureIgnoreCase;
77 public static StringComparer Ordinal {
78 get { return ordinal; }
81 public static StringComparer OrdinalIgnoreCase {
82 get { return ordinalIgnoreCase; }
86 public static StringComparer Create (CultureInfo culture, bool ignoreCase)
88 return new CultureAwareComparer (culture, ignoreCase);
91 public int Compare (object x, object y)
100 string s_x = x as string;
102 string s_y = y as string;
104 return Compare (s_x, s_y);
107 IComparable ic = x as IComparable;
109 throw new ArgumentException ();
111 return ic.CompareTo (y);
114 public new bool Equals (object x, object y)
118 if (x == null || y == null)
121 string s_x = x as string;
123 string s_y = y as string;
125 return Equals (s_x, s_y);
130 public int GetHashCode (object o)
133 throw new ArgumentNullException("o");
135 string s = o as string;
136 return s == null ? o.GetHashCode (): GetHashCode(s);
139 public abstract int Compare (string x, string y);
140 public abstract bool Equals (string x, string y);
141 public abstract int GetHashCode (string s);
145 class CultureAwareComparer : StringComparer
147 readonly bool _ignoreCase;
148 readonly CompareInfo _compareInfo;
150 public CultureAwareComparer (CultureInfo ci, bool ignore_case)
152 _compareInfo = ci.CompareInfo;
153 _ignoreCase = ignore_case;
156 public override int Compare (string x, string y)
158 CompareOptions co = _ignoreCase ? CompareOptions.IgnoreCase :
160 return _compareInfo.Compare (x, y, co);
163 public override bool Equals (string x, string y)
165 return Compare (x, y) == 0;
168 public override int GetHashCode (string s)
171 throw new ArgumentNullException("s");
173 CompareOptions co = _ignoreCase ? CompareOptions.IgnoreCase :
175 return _compareInfo.GetSortKey (s, co).GetHashCode ();
180 internal class OrdinalComparer : StringComparer
182 public OrdinalComparer (bool ignoreCase)
184 _ignoreCase = ignoreCase;
187 public override int Compare (string x, string y)
190 return String.CompareOrdinal (x, y);
192 // copied from String.CompareOrdinal()
199 else if (y == null) {
203 int max = x.Length > y.Length ? y.Length : x.Length;
204 for (int i = 0; i < max; i++) {
207 char xc = Char.ToUpperInvariant (x [i]);
208 char yc = Char.ToUpperInvariant (y [i]);
212 return max < x.Length ? -1 :
213 max == y.Length ? 0 : 1;
216 public override bool Equals (string x, string y)
220 return Compare (x, y) == 0;
223 public override int GetHashCode (string s)
226 return s.GetHashCode ();
228 return s.GetCaseInsensitiveHashCode ();
231 readonly bool _ignoreCase;