刚开始学习java不久,总会遇到字符串处理的问题,蒙的不行不行的。后来决定认真的看下String在创建的时候在堆栈中是如何进行的,只是我自己菜鸟级别的理解,有什么问题希望大家指出来。
那么首先看以下8个声明:
1 String str1 = new String("hello"); 2 String str2 = new String("hello"); 3 String str3 = "hello"; 4 String str4 = "hello"; 5 String str5 = "world"; 6 String str6 = "helloworld"; 7 String str7 = str4+str5; 8 String str8 = "hello"+"world";
String是一个final类,在每一个会修改String值的方法,都会创建一个全新的String对象。那么在初始化String时分为两类,一种是当做对象初始化(new出来的),另一种当做是基本类型,直接赋值的。
当初始化String对象时,首先会在堆内存查询是否存在要添加的对象是否存在,如果存在,则新声明的引用就会指向在堆中已存在的对象;如果不存在,则在堆中新开辟内存,存储对象并将引用指向它,无论堆中是否存在相同值的对象,都会在栈内存中新开辟一块内存存储新的引用。
当初始化基本类型的String时,同样首先会查询堆内存,但这里查询要分为是否有相同值的对象还是相同值的基本类型。如果只有相同值的对象,则在栈中新开辟内存存储新的引用,并指向堆中相同值的对象;如果堆中有相同值的基本类型或者对象和基本类型都有,则新的引用存放在原有基本类型的引用相同的栈内存中,也就是两个基本类型引用的栈内存地址是相同的,引用指向堆中的唯一相同值的对象;如果没有相同值的对象或基本类型,则在堆中新开辟内存存储基本类型,并在栈中新开辟内存存储新引用。
那么由此看来,堆中是不会存放两个完全相同的字符串的。上面8个String的初始化在堆栈内存分配如下图:
“==”在String类型比较的是引用在栈内存存放的地址是否相等,.equals()是比较引用在栈内存中的值是否相等,也就是引用指向的对象地址是否相等。这里可以用System.identityHashCode(s)查询引用s的地址,用s.hashCode()查询s的值。
有了以上的分析,再来看下面的问题,好像就不是很蒙了:
1 System.out.println(str1 == str2);//false,显然两者在栈内存中的地址是不同的 2 System.out.println(str1.equals(str2));//true,两者对应的是堆中同一对象 3 System.out.println(str2 == str3);//false,同样基本类型和对象两种创建方法在栈中分配不同内存 4 System.out.println(str2.equals(str2));//true,只要值一样,.equals()就是一样的,因为指向堆中同一对象 5 System.out.println(str3 == str4);//true,相同值的基本类型在栈中分配同一内存 6 System.out.println(str4 == str5);//false,看图吧... 7 System.out.println(str6 == str4+str5);//false,String引用相加时,相当于新创建了一个对象,那么肯定是新开辟栈内存 8 System.out.println(str6 == "hello"+"world");//true,String基本类型相加,还是创建一个基本类型,那么就相当于声明了一个新的String类型"helloworld",会与已存在的基本类型引用分配相同的栈内存