Java中比较两个字符串是否相等的问题
Java中,关于比较两个字符串是否相等的问题,经常会出现。下面将分析使用 ==(注意:Java中 = 是赋值运算符,== 是比较是否相等) 和 equals()方法 来比较两个字符串相等的区别:
简单一句话,==比较的是两个字符串的地址是否为相等(同一个地址),equals()方法比较的是两个字符串对象的内容是否相同(当然,若两个字符串引用同一个地址,使用equals()比较也返回true)。
字符串比较之所以看起来复杂,是因为值类型和引用类型两种数据类型的区别:值类型,例如 int 类型,当定义并初始化两个 int 类型的变量,int i = 1;int j = i,这两个变量 i , j 存放的的就是1这个值,使用 if(i == j)判断时,自然返回 i 和 j 是相等的 。引用类型,例如字符串类型, 当定义并初始化两个字符串类型的变量,String s = "abc"; String s2 = s; 实际上是只创建了一个String类型的对象,该对象的值为abc,而 s 和 s2 都引用这一个对象。(关于值类型和引用类型的内容稍后会更新)
基础篇:
下面将举出两个例子,分析两种情况下的字符串比较问题:
例1. 使用new关键字声明两个String类型的变量
1 String s1 = new String("abc"); 2 String s2 = new String("abc"); 3 4 //分别使用.equals()和==进行比较 5 if(s1.equals(s2)) 6 { 7 System.out.println("可以使用 equals 来比较"); 8 }else 9 { 10 System.out.println("不可以使用 equals 来比较"); 11 } 12 if(s1 == s2) 13 { 14 System.out.println("可以使用== 来比较"); 15 }else 16 { 17 System.out.println("不可以使用== 来比较"); 18 }
运行程序发现,用equals比较返回true,用 == 比较返回false。原因如下:
因为 == 比较的是两个字符串对象的地址是否相同(是否为同一个地址),当使用new关键字创建一个对象的时候,该对象单独占据一块存储空间,存放llg这个字符串的值。所以s1 s2两个字符串虽然值相同,但是存储的地址不是一个地址,例如两个人都叫l“李四”但是他们的住址不在一个地方。当使用==来比较的时候,比较的是两个字符串的地址是否是同一个,所以返回false。但是使用equals()方法比较这两个字符串,将会比较两个字符串的值是否相同,所以返回true。
例2. 正常情况下的字符串声明并定义。
//声明并初始化两个赋值一样的字符串 String s1 = "abc"; String s2 = "abc"; //使用.equals()方法比较两个字符串是否相等 if(s1.equals(s2)) { System.out.println("可以使用 equals 来比较"); }else { System.out.println("不可以使用 equals 来比较"); } //使用 == 比较两个字符串是否相等 if(s1 == s2) { System.out.println("可以使用 == 来比较"); }else { System.out.println("不可以使用 == 来比较"); }
程序经过运行,发现两个方式都返回true。那么这是为什么呢?原因如下:
一般我们认为,栈中存放基本类型的变量和对象的引用变量,而new出来的对象和数组存放在堆中。然而除此之外还有一块区域叫做常量池。像我们通常想String s1 = "abc"; 这样申明的字符串对象,其值就是存储在常量池中。当我们创建String s1 = "abc"这样一个对象之后,"abc"就存储到了常量池中,当我们创建引用String s2 = "abc" 的时候,Java底层会优先在常量池中查找是否存在"abc",如果存在则让s2指向这个值,不会重新创建。所以使用 == 比较时,因为地址相同,所以比较的结果是true。
那么问题回来了,既然是有常量池,是不是字符串如果不是new出来的(一般也不会这么new,声明字符串变量一般像例2这么写),那么这些字符串都是从常量池引用同一个对象,所以用 == 比较就行了?答案是可以。