我最近接触了一个大型代码库,并注意到所有字符串比较都使用string . equals()而不是==
你认为这是什么原因?
我最近接触了一个大型代码库,并注意到所有字符串比较都使用string . equals()而不是==
你认为这是什么原因?
当前回答
这篇文章有一篇文章,你可能会觉得很有趣,引用了乔恩·斯基特的话。看起来用途是差不多的。
Jon Skeet指出,实例Equals的性能“在字符串较短时略好——随着字符串长度的增加,这种差异变得完全不重要。”
其他回答
这两种方法在功能上是相同的——它们比较值。 正如MSDN上写的那样:
关于字符串。Equals方法-确定该实例和 另一个指定的String对象具有相同的值。(http://msdn.microsoft.com/en-us/library/858x0yyx.aspx) About ==—虽然string是引用类型,但相等操作符(==和 !=)的定义是为了比较字符串对象的值,而不是 参考文献这使得字符串相等性的测试更加直观。(http://msdn.microsoft.com/en-en/library/362314fe.aspx)
但如果你的一个字符串实例是空的,这些方法的工作方式是不同的:
string x = null;
string y = "qq";
if (x == y) // returns false
MessageBox.Show("true");
else
MessageBox.Show("false");
if (x.Equals(y)) // returns System.NullReferenceException: Object reference not set to an instance of an object. - because x is null !!!
MessageBox.Show("true");
else
MessageBox.Show("false");
弦之间有实际的区别。等于和==
bool result = false;
object obj = "String";
string str2 = "String";
string str3 = typeof(string).Name;
string str4 = "String";
object obj2 = str3;
// Comparision between object obj and string str2 -- Com 1
result = string.Equals(obj, str2);// true
result = String.ReferenceEquals(obj, str2); // true
result = (obj == str2);// true
// Comparision between object obj and string str3 -- Com 2
result = string.Equals(obj, str3);// true
result = String.ReferenceEquals(obj, str3); // false
result = (obj == str3);// false
// Comparision between object obj and string str4 -- Com 3
result = string.Equals(obj, str4);// true
result = String.ReferenceEquals(obj, str4); // true
result = (obj == str4);// true
// Comparision between string str2 and string str3 -- Com 4
result = string.Equals(str2, str3);// true
result = String.ReferenceEquals(str2, str3); // false
result = (str2 == str3);// true
// Comparision between string str2 and string str4 -- Com 5
result = string.Equals(str2, str4);// true
result = String.ReferenceEquals(str2, str4); // true
result = (str2 == str4);// true
// Comparision between string str3 and string str4 -- Com 6
result = string.Equals(str3, str4);// true
result = String.ReferenceEquals(str3, str4); // false
result = (str3 == str4);// true
// Comparision between object obj and object obj2 -- Com 7
result = String.Equals(obj, obj2);// true
result = String.ReferenceEquals(obj, obj2); // false
result = (obj == obj2);// false
添加表
obj "String" {1#} object {string}
str2 "String" {1#} string
str3 "String" {5#} string
str4 "String" {1#} string
obj2 "String" {5#} object {string}
现在看{1#}和{5#}
Obj, str2, str4和obj2引用相同。
Obj和obj2是对象类型,其他是字符串类型
结论:
com1: result = (obj == str2);// true compares object and string so performs a reference equality check obj and str2 point to the same reference so the result is true com2: result = (obj == str3);// false compares object and string so performs a reference equality check obj and str3 point to the different references so the result is false com3: result = (obj == str4);// true compares object and string so performs a reference equality check obj and str4 point to the same reference so the result is true com4: result = (str2 == str3);// true compares string and string so performs a string value check str2 and str3 are both "String" so the result is true com5: result = (str2 == str4);// true compares string and string so performs a string value check str2 and str4 are both "String" so the result is true com6: result = (str3 == str4);// true compares string and string so performs a string value check str3 and str4 are both "String" so the result is true com7: result = (obj == obj2);// false - compares object and object so performs a reference equality check - obj and obj2 point to the different references so the result is false
我想补充一点,还有一点不同。这与Andrew发布的内容有关。
这也与在我们的软件中发现一个非常烦人的bug有关。请参阅下面的简化示例(我还省略了空检查)。
public const int SPECIAL_NUMBER = 213;
public bool IsSpecialNumberEntered(string numberTextBoxTextValue)
{
return numberTextBoxTextValue.Equals(SPECIAL_NUMBER)
}
这将编译并总是返回false。而下面将给出一个编译错误:
public const int SPECIAL_NUMBER = 213;
public bool IsSpecialNumberEntered(string numberTextBoxTextValue)
{
return (numberTextBoxTextValue == SPECIAL_NUMBER);
}
我们不得不解决一个类似的问题,有人使用Equals比较不同类型的枚举。在意识到这是错误的原因之前,你会阅读很多次。特别是当SPECIAL_NUMBER的定义不在问题区域附近时。
这就是为什么我真的反对在不必要的情况下使用等号。你失去了一点类型安全。
==和String之间有一个微妙但非常重要的区别。=方法:
class Program
{
static void Main(string[] args)
{
CheckEquality("a", "a");
Console.WriteLine("----------");
CheckEquality("a", "ba".Substring(1));
}
static void CheckEquality<T>(T value1, T value2) where T : class
{
Console.WriteLine("value1: {0}", value1);
Console.WriteLine("value2: {0}", value2);
Console.WriteLine("value1 == value2: {0}", value1 == value2);
Console.WriteLine("value1.Equals(value2): {0}", value1.Equals(value2));
if (typeof(T).IsEquivalentTo(typeof(string)))
{
string string1 = (string)(object)value1;
string string2 = (string)(object)value2;
Console.WriteLine("string1 == string2: {0}", string1 == string2);
}
}
}
产生如下输出:
value1: value2: value1 == value2: True value1.Equals (value2):真的 string1 == string2: True ---------- value1: value2: value1 == value2: False value1.Equals (value2):真的 string1 == string2: True
可以看到==运算符对两个明显相等的字符串返回false。为什么?因为泛型方法中使用的==运算符被解析为System定义的op_equal方法。对象(该方法在编译时拥有的T的唯一保证),这意味着它是引用相等而不是值相等。
当有两个值类型为System时。String,则==具有值相等语义,因为编译器将==解析为System.String。op_equal代替System.Object.op_equal。
所以为了安全起见,我几乎总是使用String。我总是得到我想要的值相等语义。
如果其中一个值为null,为了避免nullreferenceexception,我总是使用静态String。=方法:
bool true = String.Equals("a", "ba".Substring(1));
字符串。Equals确实提供了重载来处理区分大小写和区域性的比较。如果您的代码没有使用这些,开发人员可能只是习惯于Java,在那里(正如Matthew所说),您必须使用. equals方法来进行内容比较。