java中String s="abc"及String s=new String("abc")详解

本文转自:https://www.cnblogs.com/quyanhui/p/3781663.html
本人书写这个纯粹只是为了便于自己查找,开放是为了方便各位指出我的错误。

1、java中的堆与栈都是Java中用来RAM内存中使用的,Java自动通过GC自动管理堆栈,无需程序员管理。
2、存储在栈的对象,变量,生命周期都是已知的程序结束对象就会被死亡,优势就是存取速度快。但相比于堆,它可存储数据的大小,生命周期都是固定的,不够灵活。 另外,栈的数据可以共享。堆的优势便是可以动态的分配大小,生存周期也不需要告诉编译器(但是容易引起GC问题),GC会自动处理这个。但是缺点就是,由于是在运行中动态的分配内存,所以存取速度会慢一些。

==是判断两个对象是否是同一个对象 
equals是进行值的判断 
String   a   =   new   String( "aaa "); 
String   b   =   new   String( "a "); 
b   +=   "aa "; 
则   a==b   //错误 
      a.equals(b)//正确

除了String和封装器,equals()和“==”没什么区别 
但String和封装器重写了equals(),所以在这里面,equals()指比较字符串或封装对象对应的原始值是否相等, "== "是比较两个对象是否为同一个对象

首先我们看一下Java中变量的的语义:
Java中的语义定义有两种,原始类型的定义是值(value)定义,也就是说,你给一个原始类型赋值,就改变了这个数据本身。第二种对象类型,对象类型的变量是引用语义,也就是说你给一个对象类型赋值只是让它指向了一个对象(对象地址or地址),但是不会改变原来引用对象的值。

《String的特性》
1、String类是final的,不可被继承的。
2、String类本质是字符数组char[],并且其值不可变。
3、String类对象有个特殊的创建方式,就是直接指定String s = “abc”,“abc”就代表一个字符串对象,而s就是“abc”的地址(我理解为s指向“abc”的地址),也叫做“abc”对象的引用。
4、String对象可以通过“+”串联,串联后会形成新的字符串对象。
5、String运行时会维护一个String Pool(String池),JavaDoc翻译很模糊“字符串缓冲区”。String池用来存放运行时产生的各种字符串,并且池中的字符串不可以重复。而一般对象不存在这个缓存区,并且创建的对象仅仅存在于方法的堆栈中。
6、创建字符串的方式很多,归纳起来分为三种:
1)使用new关键字创建,例如:String s = new String(“abc”);(注意这里是创建了两个对象)
2)直接指定 s2 = “abc”;
3)使用串联生成新的字符串。例如 String s3 = “ab” + “cd”;

《String对象的创建》
String对象的创建也有很多门道,关键就是明白其中的原理。
原理1:当使用任何方式来创建一个字符串对象s=X的时候,Java运行时(运行中JVM)会拿着这个X去String池中查找是否存在相同的对象,如果不存在,则在池中添加,如果存在,怎不添加。

原理2;Java中,只要使用new关键字创建对象,则一定会(在堆中或者栈中)创建一个新的对象。

原理3:直接指定或者使用纯字符串串联来创建对象,则仅仅检查维护Stirng池中字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。

原理4:使用包含变量的方式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区中创建新的字符串对象。

《不可变类》
Java为了提高效率,对String类型进行特殊处理–为String类型提供了字符串池
定义一个String类型的变量有两种方式:
String s = “tom”;
String s = new(“tom”);

如果你使用第一种方式,那么当你申明一个内容是“tom”的String时,他将会使用串池里原来的那个内存(地址),而不会重新分配内存,也就是说String s = “tom“;将会指向同一块内存。而如果使用第二种方式,不管串池里面有没有,都会在堆中开辟一块新的内存,创建一个新的对象。

另外关于String类型是不可改变的问题:String类型是不可改变的,也就是说,当你想改变一个String对象的时候,例如Stirng s = “tom”;那么虚拟机不会改变原来的对象,而是会生成一个新的String对象,然后让s取直向它,如果原来的那个“tom”没有任何对象去引用它,GC将会回收它。

最后,关于String堆栈的问题
String s = new String();分析堆栈,是先定义s,还是先new String()。
1、String str1 = “abc”;
System.out.println(str1 == “abc”);
步骤:
1)栈中开辟一块空间存放引用str1;
2)String池中开辟一块空间,存放String常量”abc”;
3)引用str1指向池中String常量”abc”。
4)str1所指代的地址即常量”abc”所在地址,输出为true。

2、String str2 = new String(“abc”);
System.out.println(str2 == “abc”);\
步骤:
1)栈中开辟一块空间存放引用str2。
2)堆中开辟一块空间存放一个新建的String对象”abc”。
3)引用str2指向堆中新创建的对象”abc”。
4)str2所指代的对象地址为堆中地址,而常量”abc”地址在池中,输出为false。

3.Sting str3 = new String(“abc”);
System.out.println(str3 == str2);
步骤:
1)栈中开辟一块空间存放引用str3。
2)堆中开辟一块新空间存放另外一个(不同于str2所指)新建的String对象。
3)引用str3指向另外新建的那个String对象。
4)str2和str3分别指向堆中不同的String对象,地址也不相同,输出为false。

4.String str4 = “a” + “b”;
System.out.println(str4 == “ab”);
步骤:
1)栈中开辟一块空间存放引用str4。
2)根据编译器合并已知量的优化功能,池中开辟一块空间,存放合并后的String常量”ab”。
3)引用str4指向池中的常量”ab”。
4)str4所指即池中常量”ab”,输出为true。

5.final String s = “a”;//注意这个s用final修饰,相当于常量。
String str5 = s +”b”;
System.out.println(str5 == “ab”);
步骤:
同4。

6.String s1 = “a”;
String s2 = “b”;
String str6 = s1 + s2;
System.out.println(str6 == “ab”);
步骤:
1)栈中开辟一块空间存放引用s1,s1指向池中String常量”a”,
2)栈中开辟一块空间存放引用s2,s2指向池中String常量”b”,
3)栈中开辟一块空间存放引用str6,
4)s1+s2通过StringBuilder的最后一步toString()方法还原一个新的String对象”ab”,因此堆中开辟一块空间存放对象。
5)引用str6指向堆中(s1 + s2)所还原的新String对象
6)str6指向的对象在堆中,而常量”ab”在池中,所有输出false。

7.String str7= “abc”.substring(0, 2);
步骤:
1)栈中开辟一块空间存放引用str7,
2)substring()方法还原一个新的String对象”ab”(不同于str6所指),堆中开辟一块空间存放此对象。
3)引用str7指向堆中新创建的String对象

8.String str8 = “abc”.toUpperCase();
步骤:
1)栈中开辟一块空间存放引用str8,
2)toUpperCase()方法还原一个新的String对象”ABC”,池中并未开辟新的空间存放String常量”ABC”
3)引用str8指向堆中的新String对象

9.String s = “abc”;
String s1 = s;
System.out.println(s1 == “abc”);
s = s + “hello”;
System.out.println(s1==”abc”);
System.out.println(s==”abc”);
步骤:
1)栈中开辟一块空间存放s;
2)String池中开辟一块空间用于存放”abc”;栈中开辟一块空间存放变量s1;
3)系统输出true,,在堆中开辟一块空间用于存放”abchello”;
4)引用s指向堆中”abchello”;
5)系统输出true,然后在输出false。

阅读更多

String s = "abc" 创建了....

06-19

关于String str = "abc"的内部工作。Java内部将此语句转化为以下几个步骤:rnrn  (1)先定义一个名为str的对String类的对象引用变量:String str;rnrn  (2)在栈中查找有没有存放值为"abc"的地址,如果没有,则开辟一个存放字面值为"abc"的地址,接着创建一个新的String类的对象o,并将o的字符串值指向这个地址,而且在栈中这个地址旁边记下这个引用的对象o。如果已经有了值为"abc"的地址,则查找对象o,并返回o的地址。rnrn  (3)将str指向对象o的地址。rnrn  值得注意的是,一般String类中字符串值都是直接存值的。但像String str = "abc";这种场合下,其字符串值却是保存了一个指向存在栈中数据的引用!rnrn  为了更好地说明这个问题,我们可以通过以下的几个代码进行验证。rnrnString str1 = "abc";rnString str2 = "abc";rnSystem.out.println(str1==str2); //true rnrn  注意,我们这里并不用str1.equals(str2);的方式,因为这将比较两个字符串的值是否相等。==号,根据JDK的说明,只有在两个引用都指向了同一个对象时才返回真值。而我们在这里要看的是,str1与str2是否都指向了同一个对象。rn结果说明,JVM创建了两个引用str1和str2,但只创建了一个对象,而且两个引用都指向了这个对象。rnrn  我们再来更进一步,将以上代码改成:rnrnString str1 = "abc";rnString str2 = "abc";rnstr1 = "bcd";rnSystem.out.println(str1 + "," + str2); //bcd, abcrnSystem.out.println(str1==str2); //false rnrn  这就是说,赋值的变化导致了类对象引用的变化,str1指向了另外一个新对象!而str2仍旧指向原来的对象。上例中,当我们将str1的值改为"bcd"时,JVM发现在栈中没有存放该值的地址,便开辟了这个地址,并创建了一个新的对象,其字符串的值指向这个地址。rnrn  事实上,String类被设计成为不可改变(immutable)的类。如果你要改变其值,可以,但JVM在运行时根据新值悄悄创建了一个新对象,然后将这个对象的地址返回给原来类的引用。这个创建过程虽说是完全自动进行的,但它毕竟占用了更多的时间。在对时间要求比较敏感的环境中,会带有一定的不良影响。rnrn  再修改原来代码:rnrnString str1 = "abc";rnString str2 = "abc";rnrnstr1 = "bcd";rnrnString str3 = str1;rnSystem.out.println(str3); //bcdrnrnString str4 = "bcd";rnSystem.out.println(str1 == str4); //true rnrn  str3这个对象的引用直接指向str1所指向的对象(注意,str3并没有创建新对象)。当str1改完其值后,再创建一个String的引用str4,并指向因str1修改值而创建的新的对象。可以发现,这回str4也没有创建新的对象,从而再次实现栈中数据的共享。rnrn  我们再接着看以下的代码。rnrnString str1 = new String("abc");rnString str2 = "abc";rnSystem.out.println(str1==str2); //false rnrn  创建了两个引用。创建了两个对象。两个引用分别指向不同的两个对象。rnrnString str1 = "abc";rnString str2 = new String("abc");rnSystem.out.println(str1==str2); //false rnrn  创建了两个引用。创建了两个对象。两个引用分别指向不同的两个对象。rnrn  以上两段代码说明,只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。rn

没有更多推荐了,返回首页