5 // Patrik Torstensson (patrik.torstensson@labs2.com)
6 // Jeffrey Stedfast (fejj@ximian.com)
7 // Dan Lewis (dihlewis@yahoo.co.uk)
9 // (C) 2001 Ximian, Inc. http://www.ximian.com
14 using System.Collections;
15 using System.Globalization;
16 using System.Runtime.CompilerServices;
19 public sealed class String : IConvertible, IComparable, ICloneable, IEnumerable {
22 public static readonly String Empty = "";
24 [CLSCompliant(false), MethodImplAttribute(MethodImplOptions.InternalCall)]
25 unsafe public extern String(char *value);
27 [CLSCompliant(false), MethodImplAttribute(MethodImplOptions.InternalCall)]
28 unsafe public extern String(char *value, int sindex, int length);
30 [CLSCompliant(false), MethodImplAttribute(MethodImplOptions.InternalCall)]
31 unsafe public extern String(sbyte *value);
33 [CLSCompliant(false), MethodImplAttribute(MethodImplOptions.InternalCall)]
34 unsafe public extern String(sbyte *value, int sindex, int length);
36 [CLSCompliant(false), MethodImplAttribute(MethodImplOptions.InternalCall)]
37 unsafe public extern String(sbyte *value, int sindex, int length, Encoding enc);
39 [MethodImplAttribute(MethodImplOptions.InternalCall)]
40 public extern String(char [] val, int sindex, int length);
42 [MethodImplAttribute(MethodImplOptions.InternalCall)]
43 public extern String(char [] val);
45 [MethodImplAttribute(MethodImplOptions.InternalCall)]
46 public extern String(char c, int count);
48 [MethodImplAttribute(MethodImplOptions.InternalCall)]
49 public extern override int GetHashCode();
51 public static bool Equals(String str1, String str2) {
52 if ((Object) str1 == (Object) str2)
55 if (null == (Object) str1 || null == (Object) str2)
58 if (str1.length != str2.length)
61 return InternalEquals(str1, str2);
64 public static bool operator == (String str1, String str2) {
65 return Equals(str1, str2);
68 public static bool operator != (String str1, String str2) {
69 return !Equals(str1, str2);
72 public override bool Equals(Object obj) {
79 return InternalEquals(this, (String) obj);
82 public bool Equals(String value) {
86 if (length != value.length)
89 return InternalEquals(this, value);
92 [IndexerName("Chars")]
93 public extern char this[int index] {
94 [MethodImplAttribute(MethodImplOptions.InternalCall)]
98 public Object Clone() {
102 public TypeCode GetTypeCode () {
103 return TypeCode.String;
106 public void CopyTo(int sindex, char[] dest, int dindex, int count) {
107 // LAMESPEC: should I null-terminate?
110 throw new ArgumentNullException();
112 if (sindex < 0 || dindex < 0 || count < 0)
113 throw new ArgumentOutOfRangeException ();
115 if (sindex + count > Length)
116 throw new ArgumentOutOfRangeException ();
118 if (dindex + count > dest.Length)
119 throw new ArgumentOutOfRangeException ();
121 InternalCopyTo(sindex, dest, dindex, count);
124 public char[] ToCharArray() {
125 return ToCharArray(0, length);
128 public char[] ToCharArray(int sindex, int length) {
129 if (sindex < 0 || length < 0 || sindex + length > this.length)
130 throw new ArgumentOutOfRangeException ();
132 char [] tmp = new char[length];
134 InternalCopyTo(sindex, tmp, 0, length);
139 public String [] Split(params char [] separator) {
140 return Split(separator, Int32.MaxValue);
143 public String[] Split(char[] separator, int count) {
144 if (null == separator) {
145 separator = WhiteChars;
149 throw new ArgumentOutOfRangeException ();
152 return new String[1] { ToString() };
154 return InternalSplit(separator, count);
157 public String Substring (int sindex) {
158 if (sindex < 0 || sindex > this.length) {
159 throw new ArgumentOutOfRangeException();
162 string tmp = InternalAllocateStr(this.length - sindex);
163 InternalStrcpy(tmp, 0, this, sindex, length - sindex);
168 public String Substring (int sindex, int length) {
169 if (length < 0 || sindex < 0 || sindex + length > this.length) {
170 throw new ArgumentOutOfRangeException();
176 string tmp = InternalAllocateStr(length);
177 InternalStrcpy(tmp, 0, this, sindex, length);
182 private static readonly char[] WhiteChars = { (char) 0x9, (char) 0xA, (char) 0xB, (char) 0xC, (char) 0xD, (char) 0x20, (char) 0xA0, (char) 0x2000, (char) 0x2001, (char) 0x2002, (char) 0x2003, (char) 0x2004, (char) 0x2005,
183 (char) 0x2006, (char) 0x2007, (char) 0x2008, (char) 0x2009, (char) 0x200A, (char) 0x200B, (char) 0x3000, (char) 0xFEFF };
185 public String Trim(params char[] chars) {
186 if (null == chars || chars.Length == 0)
189 return InternalTrim(chars, 0);
192 public String TrimStart(params char[] chars) {
193 if (null == chars || chars.Length == 0)
196 return InternalTrim(chars, 1);
199 public String TrimEnd(params char[] chars) {
200 if (null == chars || chars.Length == 0)
203 return InternalTrim(chars, 2);
206 public static int Compare(String s1, String s2) {
207 return Compare(s1, s2, false);
210 public static int Compare(String s1, String s2, bool inCase) {
216 } else if (null == s2)
219 return InternalCompare(s1, 0, s2, 0, Math.Max(s1.length, s2.length), inCase);
223 public static int Compare(String s1, String s2, bool inCase, CultureInfo culture) {
224 return Compare(s1, s2, inCase);
227 public static int Compare(String s1, int i1, String s2, int i2, int length) {
228 return Compare(s1, i1, s2, i2, length, false);
231 public static int Compare(String s1, int i1, String s2, int i2, int length, bool inCase) {
237 } else if (null == s2)
240 if (length < 0 || i1 < 0 || i2 < 0)
241 throw new ArgumentOutOfRangeException ();
243 if (i1 > s1.length || i2 > s2.length)
244 throw new ArgumentOutOfRangeException ();
249 return InternalCompare(s1, i1, s2, i2, length, inCase);
253 public static int Compare(String s1, int i1, String s2, int i2, int length, bool inCase, CultureInfo culture) {
254 return Compare(s1, i1, s2, i2, length, inCase);
257 public int CompareTo(Object value) {
261 if (!(value is String))
262 throw new ArgumentException();
264 return String.Compare(this, (String) value, false);
267 public int CompareTo(String str) {
271 return Compare(this, str, false);
274 public static int CompareOrdinal(String s1, String s2) {
275 if (null == s1 || null == s2) {
276 if ((Object)s1 == (Object) s2) {
280 return (s1 == null) ? -1 : 1;
283 return InternalCompare(s1, 0, s2, 0, Math.Max(s1.length, s2.length), false);
286 public static int CompareOrdinal(String s1, int i1, String s2, int i2, int length) {
287 if (null == s1 || null == s2) {
288 if ((Object)s1 == (Object) s2) {
292 return (s1 == null) ? -1 : 1;
295 if (i1 < 0 || i2 < 0 || length < 0)
296 throw new ArgumentOutOfRangeException ();
298 if (i1 > s1.length || i2 > s2.length)
299 throw new ArgumentOutOfRangeException ();
301 return InternalCompare(s1, i1, s2, i2, length, false);
304 public bool EndsWith(String value) {
306 throw new ArgumentNullException();
308 if (value.length > this.length) {
312 return (0 == Compare(this, length - value.length, value, 0, value.length));
315 public int IndexOfAny(char [] arr) {
317 throw new ArgumentNullException();
319 return InternalIndexOfAny(arr, 0, this.length);
322 public int IndexOfAny(char [] arr, int sindex) {
324 throw new ArgumentNullException();
325 if (sindex < 0 || sindex >= this.length)
326 throw new ArgumentOutOfRangeException();
328 return InternalIndexOfAny(arr, sindex, this.length - sindex);
331 public int IndexOfAny(char [] arr, int sindex, int count) {
333 throw new ArgumentNullException();
334 if (sindex < 0 || count < 0 || sindex + count > this.length)
335 throw new ArgumentOutOfRangeException ();
337 return InternalIndexOfAny(arr, sindex, count);
340 public int IndexOf(char value) {
341 return InternalIndexOf(value, 0, this.length);
344 public int IndexOf(String value) {
345 return IndexOf(value, 0, this.length);
348 public int IndexOf(char value, int sindex) {
349 if (sindex < 0 || sindex >= this.length) {
350 throw new ArgumentOutOfRangeException();
353 return InternalIndexOf(value, sindex, this.length - sindex);
356 public int IndexOf(String value, int sindex) {
357 return IndexOf(value, sindex, this.length - sindex);
360 public int IndexOf(char value, int sindex, int count) {
361 if (sindex < 0 || count < 0 || sindex + count > this.length)
362 throw new ArgumentOutOfRangeException ();
364 if (sindex == 0 && this.length == 0)
367 return InternalIndexOf(value, sindex, count);
370 public int IndexOf(String value, int sindex, int count) {
372 throw new ArgumentNullException();
374 if (sindex < 0 || count < 0 || sindex + count > this.length)
375 throw new ArgumentOutOfRangeException ();
377 if (sindex == 0 && this.length == 0)
380 return InternalIndexOf(value, sindex, count);
383 public int LastIndexOfAny(char [] arr) {
385 throw new ArgumentNullException();
387 return InternalLastIndexOfAny(arr, this.length - 1, this.length);
390 public int LastIndexOfAny(char [] arr, int sindex) {
392 throw new ArgumentNullException();
394 if (sindex < 0 || sindex > this.length)
395 throw new ArgumentOutOfRangeException();
397 if (this.length == 0)
400 return InternalLastIndexOfAny(arr, sindex, sindex + 1);
403 public int LastIndexOfAny(char [] arr, int sindex, int count) {
405 throw new ArgumentNullException();
407 if (sindex < 0 || count < 0 || sindex > this.length || sindex - count < -1)
408 throw new ArgumentOutOfRangeException();
410 if (this.length == 0)
413 return InternalLastIndexOfAny(arr, sindex, count);
416 public int LastIndexOf(char value) {
417 return InternalLastIndexOf(value, this.length - 1, this.length);
420 public int LastIndexOf(String value) {
422 throw new ArgumentNullException();
424 if (value.length == 0)
\r
427 if (this.length == 0)
\r
430 return InternalLastIndexOf(value, this.length - 1, this.length);
433 public int LastIndexOf(char value, int sindex){
434 return LastIndexOf(value, sindex, sindex + 1);
437 public int LastIndexOf(String value, int sindex) {
438 return LastIndexOf(value, sindex, sindex + 1);
441 public int LastIndexOf(char value, int sindex, int count) {
442 if (sindex < 0 || count < 0)
\r
443 throw new ArgumentOutOfRangeException ();
\r
445 if (sindex >= this.length || sindex - count + 1 < 0)
\r
446 throw new ArgumentOutOfRangeException ();
\r
448 if (sindex == 0 && this.length == 0)
451 return InternalLastIndexOf(value, sindex, count);
454 public int LastIndexOf(String value, int sindex, int count) {
456 throw new ArgumentNullException();
458 if (sindex < 0 || sindex > this.length)
\r
459 throw new ArgumentOutOfRangeException ();
\r
461 if (count < 0 || sindex - count + 1 < 0)
\r
462 throw new ArgumentOutOfRangeException ();
\r
464 if (value.length > sindex)
\r
467 if (value == String.Empty)
\r
470 if (sindex == 0 && this.length == 0)
473 return InternalLastIndexOf(value, sindex, count);
476 public String PadLeft(int width) {
477 return PadLeft(width, ' ');
480 public String PadLeft(int width, char chr) {
482 throw new ArgumentException();
484 if (width < this.length)
485 return String.Copy(this);
487 return InternalPad(width, chr, false);
490 public String PadRight(int width) {
491 return PadRight(width, ' ');
494 public String PadRight(int width, char chr) {
496 throw new ArgumentException();
498 if (width < this.length)
499 return String.Copy(this);
501 return InternalPad(width, chr, true);
504 public bool StartsWith(String value) {
506 throw new ArgumentNullException();
508 if (this.length < value.length)
511 return (0 == Compare(this, 0, value, 0 , value.length));
515 public String Replace (char oldChar, char newChar) {
516 return InternalReplace(oldChar, newChar);
519 public String Replace(String oldValue, String newValue) {
520 if (null == oldValue)
521 throw new ArgumentNullException();
523 return InternalReplace(oldValue, newValue);
526 public String Remove(int sindex, int count) {
527 if (sindex < 0 || count < 0 || sindex + count > this.length)
528 throw new ArgumentOutOfRangeException ();
530 return InternalRemove(sindex, count);
533 public String ToLower() {
534 return InternalToLower();
537 public String ToLower(CultureInfo culture) {
538 throw new NotImplementedException();
541 public String ToUpper() {
542 return InternalToUpper();
545 public String ToUpper(CultureInfo culture) {
546 throw new NotImplementedException();
549 public override String ToString() {
553 public String ToString(IFormatProvider provider) {
557 public String Trim() {
561 public static String Format(String format, Object arg0) {
562 return Format(null, format, new Object[] {arg0});
565 public static String Format(String format, Object arg0, Object arg1) {
566 return Format(null, format, new Object[] {arg0, arg1});
569 public static String Format(String format, Object arg0, Object arg1, Object arg2) {
570 return Format(null, format, new Object[] {arg0, arg1, arg2});
573 public static string Format (string format, params object[] args) {
574 return Format (null, format, args);
577 public static string Format (IFormatProvider provider, string format, params object[] args) {
578 if (format == null || args == null)
579 throw new ArgumentNullException ();
581 StringBuilder result = new StringBuilder ();
585 while (ptr < format.length) {
586 char c = format[ptr ++];
589 result.Append (format, start, ptr - start - 1);
591 // check for escaped open bracket
593 if (format[ptr] == '{') {
604 ParseFormatSpecifier (format, ref ptr, out n, out width, out left_align, out arg_format);
605 if (n >= args.Length)
606 throw new FormatException ("Index (zero based) must be greater than or equal to zero and less than the size of the argument list.");
610 object arg = args[n];
615 else if (arg is IFormattable)
616 str = ((IFormattable)arg).ToString (arg_format, provider);
618 str = arg.ToString ();
620 // pad formatted string and append to result
622 if (width > str.length) {
623 string pad = new String (' ', width - str.length);
639 else if (c == '}' && format[ptr] == '}') {
640 result.Append (format, start, ptr - start - 1);
645 if (start < format.length)
646 result.Append (format.Substring (start));
648 return result.ToString ();
651 public static String Copy (String str) {
653 throw new ArgumentNullException ();
655 int length = str.length;
657 String tmp = InternalAllocateStr(length);
658 InternalStrcpy(tmp, 0, str);
662 public static String Concat(Object obj) {
666 return obj.ToString();
669 public static String Concat(Object obj1, Object obj2) {
676 return Concat(obj1.ToString(), obj2.ToString());
679 public static String Concat(Object obj1, Object obj2, Object obj3) {
689 return Concat(obj1.ToString(), obj2.ToString(), obj3.ToString());
693 // I can not find the "__arglist" argument on the spec
696 [CLSCompliant(false)]
697 public static String Concat (Object arg0, Object arg1, Object arg2, Object arg3, __arglist) {
698 throw new NotImplementedException();
702 public static String Concat(String s1, String s2) {
704 if (null == s2) { return String.Empty; }
708 if (null == s2) { return s1; }
710 String tmp = InternalAllocateStr(s1.length + s2.length);
712 InternalStrcpy(tmp, 0, s1);
713 InternalStrcpy(tmp, s1.length, s2);
718 public static String Concat(String s1, String s2, String s3) {
719 if (null == s1 && null == s2 && null == s3) {
723 if (null == s1) { s1 = String.Empty; }
724 if (null == s2) { s2 = String.Empty; }
725 if (null == s3) { s3 = String.Empty; }
727 String tmp = InternalAllocateStr(s1.length + s2.length + s3.length);
729 InternalStrcpy(tmp, 0, s1);
730 InternalStrcpy(tmp, s1.length, s2);
731 InternalStrcpy(tmp, s1.length + s2.length, s3);
736 public static String Concat(String s1, String s2, String s3, String s4) {
737 if (null == s1 && null == s2 && null == s3 && null == s4) {
741 if (null == s1) { s1 = String.Empty; }
742 if (null == s2) { s2 = String.Empty; }
743 if (null == s3) { s3 = String.Empty; }
744 if (null == s4) { s4 = String.Empty; }
746 String tmp = InternalAllocateStr(s1.length + s2.length + s3.length + s4.length);
748 InternalStrcpy(tmp, 0, s1);
749 InternalStrcpy(tmp, s1.length, s2);
750 InternalStrcpy(tmp, s1.length + s2.length, s3);
751 InternalStrcpy(tmp, s1.length + s2.length + s3.length, s4);
756 public static String Concat(params Object[] args) {
758 int len, i, currentpos;
761 throw new ArgumentNullException ();
763 strings = new string [args.Length];
766 foreach (object arg in args) {
767 /* use Empty for each null argument */
769 strings[i] = String.Empty;
771 strings[i] = arg.ToString ();
772 len += strings[i].length;
781 String tmp = InternalAllocateStr(len);
782 for (i = 0; i < strings.Length; i++) {
783 InternalStrcpy(tmp, currentpos, strings[i]);
784 currentpos += strings[i].length;
790 public static String Concat(params String[] values) {
791 int len, i, currentpos;
794 throw new ArgumentNullException ();
797 foreach (string value in values)
798 len += value != null ? value.length : 0;
805 String tmp = InternalAllocateStr(len);
806 for (i = 0; i < values.Length; i++) {
807 if (values[i] == null)
810 InternalStrcpy(tmp, currentpos, values[i]);
811 currentpos += values[i].length;
817 public String Insert(int sindex, String value) {
819 throw new ArgumentNullException();
821 if (sindex < 0 || sindex > this.length)
822 throw new ArgumentOutOfRangeException();
824 return InternalInsert(sindex, value);
828 public static string Intern (string str) {
830 throw new ArgumentNullException ();
832 return InternalIntern(str);
835 public static string IsInterned (string str) {
837 throw new ArgumentNullException();
839 return InternalIsInterned(str);
842 public static string Join (string separator, string [] value) {
844 throw new ArgumentNullException ();
846 return Join(separator, value, 0, value.Length);
849 public static string Join(string separator, string[] value, int sindex, int count) {
851 throw new ArgumentNullException ();
853 if (sindex + count > value.Length)
854 throw new ArgumentOutOfRangeException ();
856 if (sindex == value.Length)
859 return InternalJoin(separator, value, sindex, count);
862 bool IConvertible.ToBoolean (IFormatProvider provider) {
863 return Convert.ToBoolean (this);
866 byte IConvertible.ToByte (IFormatProvider provider) {
867 return Convert.ToByte (this);
870 char IConvertible.ToChar (IFormatProvider provider) {
871 return Convert.ToChar (this);
874 DateTime IConvertible.ToDateTime (IFormatProvider provider) {
875 return Convert.ToDateTime (this);
878 decimal IConvertible.ToDecimal (IFormatProvider provider) {
879 return Convert.ToDecimal (this);
882 double IConvertible.ToDouble (IFormatProvider provider) {
883 return Convert.ToDouble (this);
886 short IConvertible.ToInt16 (IFormatProvider provider) {
887 return Convert.ToInt16 (this);
890 int IConvertible.ToInt32 (IFormatProvider provider) {
891 return Convert.ToInt32 (this);
894 long IConvertible.ToInt64 (IFormatProvider provider) {
895 return Convert.ToInt64 (this);
898 [CLSCompliant(false)]
899 sbyte IConvertible.ToSByte (IFormatProvider provider) {
900 return Convert.ToSByte (this);
903 float IConvertible.ToSingle (IFormatProvider provider) {
904 return Convert.ToSingle (this);
906 string IConvertible.ToString (IFormatProvider format) {
910 object IConvertible.ToType (Type conversionType, IFormatProvider provider) {
911 return Convert.ToType (this, conversionType, provider);
914 [CLSCompliant(false)]
915 ushort IConvertible.ToUInt16 (IFormatProvider provider) {
916 return Convert.ToUInt16 (this);
919 [CLSCompliant(false)]
920 uint IConvertible.ToUInt32 (IFormatProvider provider) {
921 return Convert.ToUInt32 (this);
924 [CLSCompliant(false)]
925 ulong IConvertible.ToUInt64 (IFormatProvider provider) {
926 return Convert.ToUInt64 (this);
929 TypeCode IConvertible.GetTypeCode () {
930 return TypeCode.String;
939 public CharEnumerator GetEnumerator () {
940 return new CharEnumerator (this);
943 IEnumerator IEnumerable.GetEnumerator () {
944 return new CharEnumerator (this);
947 private static void ParseFormatSpecifier (string str, ref int ptr, out int n, out int width, out bool left_align, out string format) {
948 // parses format specifier of form:
954 // N = argument number (non-negative integer)
956 n = ParseDecimal (str, ref ptr);
958 throw new FormatException ("Input string was not in correct format.");
960 // M = width (non-negative integer)
962 if (str[ptr] == ',') {
963 left_align = (str[++ ptr] == '-');
967 width = ParseDecimal (str, ref ptr);
969 throw new FormatException ("Input string was not in correct format.");
976 // F = argument format (string)
978 if (str[ptr] == ':') {
980 while (str[ptr] != '}')
983 format = str.Substring (start, ptr - start);
988 if (str[ptr ++] != '}')
989 throw new FormatException ("Input string was not in correct format.");
991 catch (IndexOutOfRangeException) {
992 throw new FormatException ("Input string was not in correct format.");
996 private static int ParseDecimal (string str, ref int ptr) {
1001 if (c < '0' || '9' < c)
1004 n = n * 10 + c - '0';
1015 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1016 private extern static string InternalJoin(string separator, string[] value, int sindex, int count);
1018 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1019 private extern String InternalInsert(int sindex, String value);
1021 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1022 private extern String InternalReplace(char oldChar, char newChar);
1024 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1025 private extern String InternalReplace(String oldValue, String newValue);
1027 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1028 private extern String InternalRemove(int sindex, int count);
1030 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1031 private extern void InternalCopyTo(int sindex, char[] dest, int dindex, int count);
1033 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1034 private extern String[] InternalSplit(char[] separator, int count);
1036 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1037 private extern String InternalTrim(char[] chars, int typ);
1039 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1040 private extern int InternalIndexOf(char value, int sindex, int count);
1042 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1043 private extern int InternalIndexOf(string value, int sindex, int count);
1045 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1046 private extern int InternalIndexOfAny(char [] arr, int sindex, int count);
1048 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1049 private extern int InternalLastIndexOf(char value, int sindex, int count);
1051 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1052 private extern int InternalLastIndexOf(String value, int sindex, int count);
1054 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1055 private extern int InternalLastIndexOfAny(char [] anyOf, int sindex, int count);
1057 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1058 private extern String InternalPad(int width, char chr, bool right);
1060 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1061 private extern String InternalToLower();
1063 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1064 private extern String InternalToUpper();
1066 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1067 private extern static String InternalAllocateStr(int length);
1069 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1070 private extern static void InternalStrcpy(String dest, int destPos, String src);
1072 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1073 private extern static void InternalStrcpy(String dest, int destPos, String src, int startPos, int count);
1075 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1076 private extern static string InternalIntern(string str);
1078 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1079 private extern static string InternalIsInterned(string str);
1081 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1082 private extern static int InternalCompare(String s1, int i1, String s2, int i2, int length, bool inCase);
1084 [MethodImplAttribute(MethodImplOptions.InternalCall)]
1085 private extern static bool InternalEquals(String s1, String s2);