A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 孙利川 中级黑马   /  2012-4-1 17:33  /  1735 人查看  /  8 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

String s1 = "abc";
String s2 = new String("abc");
为什么s1在内存中有一个对象而s2在内存中则有两个对象?

8 个回复

倒序浏览
相信你看看下面的例子,就能明白了!
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指向堆中的新建的String对象"abc";
4) str2所指代的对象地址为堆中地址,而常量"abc"地址在池中,输出为false;



3. String str3 = new String("abc");
    System.out.println(str3 == str2);

步骤:
1) 栈中开辟一块空间存放引用str3;
2) 堆中开辟一块新空间存放另外一个(不同于str2所指)新建的String对象;
3) 引用str3指向另外新建的那个String对象 ;
4) str3和str2指向堆中不同的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");

步骤:
同四



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) 栈中开辟一块中间存放引用str5,
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) 栈中开辟一块空间存放引用str6,
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)Sting池中开辟一块空间用于存放"abc",栈中开辟一块空间存放变量s1;

3)系统输出true,在堆中开辟一块空间用于存放"abchello";

4)引用s指向堆中的"abchello";

5)系统输出true,然后输出false;
回复 使用道具 举报
杨浩 发表于 2012-4-1 18:16
相信你看看下面的例子,就能明白了!
String s =new String()分析堆与栈,是先定义S,还是先new string()
1.  ...

第五个例子为什么加上final与不加final,结果却截然不同
回复 使用道具 举报
String s = new String("abc")实际上是"abc"本身就是文字池中的一个对象,
在运行 new String()时,把文字池中的字符串"abc"复制到中,并把
这个对象的应用交给s,所以创建了两个String对象,一个在池中,一个在堆中。
回复 使用道具 举报
孙利川 发表于 2012-4-1 18:41
第五个例子为什么加上final与不加final,结果却截然不同

final关键字的效果是只对引用的"值"(即内存地址)有效,强迫该变量只能指向初始指向的那个对象,你改变它的引用的时候,编译就直接报错的!
至于被指向的对象内容的变化,final不做限制滴!
不过5这个例子里被指定的对象,"a"是String池中的常量,不可改变,这点要注意!
回复 使用道具 举报
孙利川 发表于 2012-4-1 18:41
第五个例子为什么加上final与不加final,结果却截然不同

final的效果,再给你一个例子,你就理解一些了。
final StringBuffer a = new StringBuffer("a");
final StringBuffer b = new StringBuffer("b");
b = a;//编译报错
a.append("c");//这个可以。
System.out.println(a);//结果是ac
回复 使用道具 举报
孙利川 发表于 2012-4-1 18:41
第五个例子为什么加上final与不加final,结果却截然不同

http://wenku.baidu.com/view/f98a589851e79b89680226fb.html
这里还有一个String类型的详解!可以更深入的了解下!嘿嘿,其实我也没想到,由String竟然扯出来好多的细节问题,我记得曾经有老师说过,做代码分析的时候,分析到内存分配的层次,才能比较好的理解。这是解决复杂怪异问题的最好办法!
回复 使用道具 举报
String s1 = "abc";这个语句声明的是一个指向对象的引用,名为“s1”,可以指向类型为String的任何对象,目前指向 "abc"这个String类型的对象。只是声明了一个只能指向String对象的引用变量。
String s2 = new String("abc");这个语句创建两个对象!一个是字符串对象,一个是指向这个字符串的引用对象。
回复 使用道具 举报
一个是字符串对象,另一个是指向这个字符串的引用对象。
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马