引用自这位朋友:http://blog.sina.com.cn/s/blog_6a6b14100100zn6r.html

首先让我们了解几个概念:

栈:由JVM分配区域,用于保存线程执行的动作和数据引用。

堆:由JVM分配的,用于存储对象等数据的区域。

常量池constant pool :在堆中分配出来的一块存储区域,用于存储显式 的String,float或者integer.这是一个特殊的共享区域,可以在内存中共享的不经常改变的东西,都可以放在这里。

进入正题:

String a = "abc";①
String b = "abc";②

使用String a = "abc";的方式,可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。
①代码执行后在Constant Pool中创建了一个值为abc的String对象,②执行时,因为Constant Pool中存在"abc"所以就不在创建新的String对象了。

String   c   =   new   String("xyz");①
String   d   =   new   String("xyz");②

让我们来看看这两句代码在内存中发生了什么,①Class被CLassLoader加载时,你的"xyz"被作为常量读入,在constant  
pool里创建了一个共享的"xyz",然后当调用到new   String("xyz")的时候,会在heap里创建这个new  
String("xyz");②由于constant   pool中存在"xyz"所以不再创建"xyz",然后创建新的new  
String("xyz")。
对于String c = new String("xyz");的代码,与String a = "abc"不同的是一概在堆中创建新对象,不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。 
程序1
String   s1   =   new   String("xyz");     //创建二个对象,一个引用 
String   s2   =   new   String("xyz");     //创建一个对象,并且以后每执行一次创建一个对象,一个引用 
程序2 
String   s3   =   "xyz";     //创建一个对象,一个引用   
String   s4   =   "xyz";     //不创建对象,只是创建一个新的引用

重要的是理解constant pool与new关键字

当调用
intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object)
方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。(无论怎样都返回池中的对象)

下面的这个例子能帮助我们更深入的理解String的存储和赋值原理

String str1 = new String("123");
String str2 = "123";

String str3 = str1.intern();
        
       System.out.println((str1 == str2) +","+ (str3 == str2));

输出 false,true
                
       String str4 = new String("234");
       String str5 = new String("234");
        
       String str6 = str4.intern();
       String str7 = str5.intern();
        
       System.out.println((str4 == str5) +","+ (str6 == str7));

输出 false,true

你知道在java中除了8中基本类型外,其他的都是类对象以及其引用。所以 "xyz "在java中它是一个String对象.对于string类对象来说他的对象值是不能修改的,也就是具有不变性。

看:

String   s= "Hello
";

s= "Java ";

String   s1= "Hello
";

String  
s2=new   String( "Hello
");

啊,s所引用的string对象不是被修改了吗?之前所说的不变性,去那里了啊?

你别着急,让我告诉你说发生了什么事情:

在jvm的工作过程中,会创建一片的内存空间专门存入string对象。我们把这片内存空间叫做string池。

String   s= "Hello ";当jvm看到
"Hello
",在string池创建string对象存储它,并将他的引用返回给s。

s= "Java ",当jvm看到 "Java
",在string池创建新的string对象存储它,再把新建的string对象的引用返回给s。而原先的 "Hello
"仍然在string池内。没有消失,他是不能被修改的。

所以我们仅仅是改变了s的引用,而没有改变他所引用的对象,因为string对象的值是不能被修改的。

String   s1= "Hello
";jvm首先在string池内里面看找不找到字符串 "Hello
",找到,返回他的引用给s1,否则,创建新的string对象,放到string池里。这里由于s= "Hello
"了,对象已经被引用,所以依据规则s和s1都是引用同一个对象。所以  
s==s1将返回true。(==,对于非基本类型,是比较两引用是否引用内存中的同一个对象)

String   s2=String( "Hello
");jvm首先在string池内里面看找不找到字符串 "Hello
",找到,不做任何事情,否则,创建新的string对象,放到string池里面。由于遇到了new,还会在内存上(不是string池里面)创建string对象存储
"Hello
",并将内存上的(不是string池内的)string对象返回给s2。所以s==s2将返回false,不是引用同一个对象。

好现在我们看题目:

String  
s  
=  
new   String( "xyz
");

首先在string池内找,找到?不创建string对象,否则创建,  
这样就一个string对象

遇到new运算符号了,在内存上创建string对象,并将其返回给s,又一个对象

所以总共是2个对象

 

一个例子:

public class Test

{

public static void main(String []
args)

{

String s1=new
String("test");//创建2个对象,一个Class和一个堆里面

String
s2="test";//创建1个对象,s2指向pool里面的"test"对象

String
s3="test";//创建0个对象,指向s2指想pool里面的那个对象

String
s4=s2;//创建0个对象,指向s2,s3指想pool里面的那个对象

String s5=new
String("test");//创建1个对象在堆里面,注意,与s1没关系

System.out.println(s2=="test");//true
s2=="test"很明显true

System.out.println(s2==s3);//true,因为指向的都是pool里面的那个"test"

System.out.println(s2==s4);//true,同上,那么s3和s4...:)

System.out.println(s1==s5);//false,很明显,false

System.out.println(s1==s2);//false,指向的对象不一样,下面再说

System.out.println(s1=="test");//false,难道s1!="tset"?下面再说

System.out.println("---------------");

s1=s2;

System.out.println(s1=="test");//true,下面说

}

}

说明:1,System.out.println(s1==s2);很明显,s2指向的对象"test"是在pool里面,而s1指向的是堆里面的"test"对象(s1指向的内存区),所以返回false.

2,System.out.println(s1=="test");s1指向的是堆里面的"test"对象(s1指向的内存区),而"test"是程序
刚刚建立的(其实是共用pool里面的那个已经创建了的"test"对象,也就是我们s2="test"时候,在pool里面创建的),所以s1指向的堆
里的"test"对象

和"test"(pool里面)并不是一样个对象,所以返回false.

3,当我们s1=s2;的时候,很明显,把s2的指给了s1,s1指向pool里面的"test",这个时候,s2也指向了pool里面的"test"对
象了,当System.out.println(s1=="test");时候,java虚拟机创建"test"对象,注意,其实没创建,和前面讲的一
样,公用s1="test"创建的"test"对象(pool里面的),所以,s1=="test"(pool里面的),同
样,s1=s2=s3=s4!

而为什么在网上都说String s=new
String("test");创建了2个对象?那可能因为它就写这么一句代码,误让人默认的认为执行代码之前并不实例任何一个String对象过

(也许
很多人不会这么想,),跟着别人或者不经思考的就说2个,斟是说存放在栈内存中专门存放String对象引用的s变量是一个对象!实在不可原谅!

所以当有面试官问起,String   s2=new   String( "Hello "); 创建几个对象时,正确答案应该回答,视情况而定,当常量池中没有“Hello”对象时是创建两个,一个存放在常量池中,另一个存放在堆(内存)中。

关于String s = new String("xyz");创建了几个字符串对象?的问题的更多相关文章

  1. String s = new String("xyz");创建了几个字符串对象?

    两个对象,一个是静态区的"xyz",一个是用new创建在堆上的对象.

  2. String s = new String(“xyz”);创建了几个字符串对象?

    两个对象,一个是静态区的"xyz",一个是用 new 创建在堆上的对象.

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

    1.   栈(stack)与堆(heap)都是Java用来在Ram中存放数据的地方.与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆. 2.   栈的优势是,存取速度比堆要快,仅次于直 ...

  4. Java String 和 new String()的区别

    Java String 和 new String()的区别 本文转自:http://www.cnblogs.com/heima-jieqi/archive/2012/04/10/2440086.htm ...

  5. [转] 请别再拿“String s = new String("xyz");创建了多少个String实例”来面试了吧

    这帖是用来回复高级语言虚拟机圈子里的一个问题,一道Java笔试题的. 本来因为见得太多已经吐槽无力,但这次实在忍不住了就又爆发了一把.写得太长干脆单独开了一帖. 顺带广告:对JVM感兴趣的同学们同志们 ...

  6. 关于String s = new String("xyz"); 创建几个对象的问题

    引用自这位朋友:http://blog.sina.com.cn/s/blog_6a6b14100100zn6r.html 你知道在java中除了8中基本类型外,其他的都是类对象以及其引用.所以 &qu ...

  7. String str = new String("xyz") 会创建几个对象

    转载:https://blog.csdn.net/andychen314/article/details/50857313 答案是 两个对象,要理解这个,就要知道string类的工作原理.下面来慢慢分 ...

  8. 请别再拿“String s = new String("xyz");创建了多少个String实例”来面试了吧---转

    http://www.iteye.com/topic/774673 羞愧呀,不知道多少人干过,我也干过,面壁去! 这帖是用来回复高级语言虚拟机圈子里的一个问题,一道Java笔试题的. 本来因为见得太多 ...

  9. String s = new String("xyz");创建了几个String Object?并作说明。

    String s = new String("xyz");创建了几个String Object?并作说明. 共产生了两个2个对象,第一个是字符串常量xyz,存储于常量池中.第二个对 ...

随机推荐

  1. 使用spring,pageHelper ,注解完成分页。

    第一步 maven 依赖pageHeler aspectj 开启aop的注解 基础工作完成! 第二步: 建立一个注解 建立注解实现,建立环绕通知 最后一步:加上 @EnablePaging 注解 就可 ...

  2. LeetCode74.搜索二维矩阵

    74.搜索二维矩阵 描述 编写一个高效的算法来判断 m x n 矩阵中,是否存在一个目标值.该矩阵具有如下特性: 每行中的整数从左到右按升序排列. 每行的第一个整数大于前一行的最后一个整数. 示例 示 ...

  3. js去掉浏览器右键点击默认事件(+vue项目开启右键行为)

    js去掉浏览器右键点击默认事件 1.阻止整个页面所有的右击事件 document.oncontextmenu = function(){ return false;} 2.特定的区域/元素 docum ...

  4. JavaScript的type属性等于text/html 例子

    在使用JavaScript标签<script>的时候,其中type最常用的就是text/javascript 其实这个type还有其他用法,下面直接给出例子: type属性为text/ht ...

  5. Spring 配置 Apache Commons Logging

    第一次用spring framework,刚配了个最简单的项目,启动出现如下错误,查了知道原来spring要依赖Apache common logging包.只需要添加到项目library中即可.可从 ...

  6. PIE SDK专题制图另存模板

    1.功能简介 在PIE SDK中,所有的制图元素.视图范围以及排版等都可以保存成一个模板,以供多次重复使用.使用模板时只需要打开该模板,加载相应数据,就可以直接出图,省去了重复制作图幅的麻烦,方便快捷 ...

  7. SSH Intro - Remove entry

    ssh-keygen -f "/home/frank/.ssh/known_hosts" -R ec2-54-222-218-195.cn-north-1.compute.amaz ...

  8. 转-------CNN图像相似度匹配 2-channel network

    基于2-channel  network的图片相似度判别 原文地址:http://blog.csdn.net/hjimce/article/details/50098483 作者:hjimce 一.相 ...

  9. (转)Unity 之 UGUI 小总结

    转自:http://www.jianshu.com/p/5b6f5022662e 开发过程中对UGUI的一个小总结. 首先从原画师拿到效果图,美术切图,拿到碎图后打成大图. 我们先来说一下图:RGBA ...

  10. 动态替换animator的研究

    http://blog.csdn.net/tonnychu/article/details/49903657 http://blog.csdn.net/tlrainty/article/details ...