java str_Java中String str=new String("a")和String str = "a"有什么区别?

本文探讨了String对象在Java中使用==和equals()进行比较的差异,通过实例说明newString()与直接赋值的区别,以及它们在HashSet中的适用性。重点在于常量池、地址值和内容比较的讲解。
摘要由CSDN通过智能技术生成

Q:

String A="ABC";

String B=new String("ABC");

这两个值,A,B 是否相等,如果都往HashSet里面放,能放下吗?

答:(a)A==B 的判断为false;

(b)A.equals(B)为true ;因为值相等,所以都往HashSet里面放不下,只能放一个

问题:==与equals()的区别:

==:比较引用类型比较的是地址值是否相同

equals:比较引用类型默认也是比较地址值是否相同,注意:String类重写了equals()方法,比较的是内容是否相同。

String A = "ABC";内存会去查找永久代(常量池) ,如果没有的话,在永久代中中开辟一块儿内存空间,把地址付给栈指针,如果已经有了"ABC"的内存,直接把地址赋给栈指针;

因此

String str1="aa";

Srting str2="aa";

String Str3="aa";

....

这样下去,str1==Str2==str3;会一直相等下去,(a) ==的判断, (b) equals()的判断;都相等,因为他们的地址都相等,因此只在常量池中有一份内存空间,地址全部相同;

而String str = new String("a");是根据"a"这个String对象再次构造一个String对象;在堆中从新new一块儿内存,把指针赋给栈,

将新构造出来的String对象的引用赋给str。 因此 只要是new String(),则,栈中的地址都是指向最新的new出来的堆中的地址,

(a)“”==“” 是判断地址的,当然不相同;

(b)至于equals,String类型重写了 equals()方法,判断值是否相等,明显相等,因此 equals 是相等的;

这是String 重写的equals:

48304ba5e6f9fe08f3fa1abda7d326ab.png

*@see #compareTo(String)

*@see #equalsIgnoreCase(String)

*/

public booleanequals(Object anObject) {

if (this ==anObject) {

return true;

}

if (anObject instanceofString) {

String anotherString =(String) anObject;

int n =value.length;

if (n ==anotherString.value.length) {

char v1[] =value;

char v2[] =anotherString.value;

int i = 0;

while (n-- != 0) {

if (v1[i] !=v2[i])

return false;

i++;

}

return true;

}

}

return false;

}

48304ba5e6f9fe08f3fa1abda7d326ab.png

48304ba5e6f9fe08f3fa1abda7d326ab.png

public classStringDemo2 {

public static voidmain(String[] args) {

String s1 = new String("hello");

String s2 = "hello";

System.out.println(s1 == s2);//false

System.out.println(s1.equals(s2));//true

}

}

**运行结果:**

> false

> true

48304ba5e6f9fe08f3fa1abda7d326ab.png

代码详解

首先,通过main()方法进栈。

然后再栈中定义一个对象s1,去堆中开辟一个内存空间,将内存空间的引用赋值给s1,“hello”是常量,然后去字符串常量池 查看是否有hello字符串对象,没有的话分配一个空间存放hello,并且将其空间地址存入堆中new出来的空间中。

在栈中定义一个对象s2,然后去字符串常量池中查看是否有”hello”字符串对象,有,直接把”hello”的地址赋值给s2.

即s1中存的是堆中分配的空间,堆中分配的空间中存的是字符串常量池中分配空间存放”hello”的空间的地址值。而s2中之间存的是字符串常量池中分配空间存放”hello”的空间的地址值。

由于s1与s2中存放的地址不同,所以输出false。因为,类String重写了equals()方法,它比较的是引用类型的 的值是否相等,所以输出true。即结果为false、true。

0e9206d7b48a679d164d1a070215a4e3.png

Demo1

48304ba5e6f9fe08f3fa1abda7d326ab.png

public classStringDemo1 {

public static voidmain(String[] args) {

String s1 = new String("hello");

String s2 = new String("hello");

System.out.println(s1 == s2);//false

System.out.println(s1.equals(s2));//true

String s3 = new String("hello");

String s4 = "hello";

System.out.println(s3 == s4);//false

System.out.println(s3.equals(s4));//true

String s5 = "hello";

String s6 = "hello";

System.out.println(s5 == s6);//true

System.out.println(s5.equals(s6));//true

}

}

48304ba5e6f9fe08f3fa1abda7d326ab.png

40ce78dc91681a7a15805d0e45235f5c.png

Demo1详解

s1~s6用equals()的比较不解释,都是比较的值,均为true。以下讲解==

s1、s2:二者均为new出来的,各自在堆中分配有空间,并各自将内存地址赋值给s1、s2。空间地址不同,==比较为false。但是各自在堆中空间中保存的值均为在字符串常量池中的同一个对象的地址。根据Demo处的图即解释不难理解。

s3、s4同上Demo出解释。

s5、s6都是在常量池中取值,二者都指向常量池中同一对象,其地址值相同,所以结果为true。

Demo2

48304ba5e6f9fe08f3fa1abda7d326ab.png

public classStringDemo4 {

public static voidmain(String[] args) {

String s1 = "hello";

String s2 = "world";

String s3 = "helloworld";

System.out.println(s3 == s1 + s2);//false

System.out.println(s3.equals((s1 + s2)));//true

System.out.println(s3 == "hello" + "world");//trueSystem.out.println(s3.equals("hello" + "world"));//true

}

}

48304ba5e6f9fe08f3fa1abda7d326ab.png

cef8c159746bd989264df2cd872eaa57.png

Demo2详解

equals()比较方法不解释,比较值,均相等,均为true。

s1与s2相加是先在字符串常量池中开一个空间,然后拼接,这个空间的地址就是s1与s2拼接后的地址。与s3的地址不同,所以输出为false。

s3与”hello”+”world”作比较,”hello”+”world”先拼接成”helloworld”,然后再去字符串常量池中找是否有”helloworld”,有,所以和s3共用一个字符串对象,则为true。

总结

String s = new String(“hello”)会创建2(1)个对象,String s = “hello”创建1(0)个对象。

注:当字符串常量池中有对象hello时括号内成立!

字符串如果是变量相加,先开空间,在拼接。

字符串如果是常量相加,是先加,然后在常量池找,如果有就直接返回,否则,就创建。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值