`
seara
  • 浏览: 654695 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

关于Java String对象创建问题解惑

阅读更多
本文为原创,如需转载,请注明作者和出处,谢谢!

先看看下面的代码
<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
publicStringmakinStrings()
{
Strings
="Fred";
s
=s+"47";
s
=s.substring(2,5);
s
=s.toUpperCase();
returns.toString();
}


问:调用makinStrings方法会创建几个String对象呢。 答案:3个


上面的方法有五条语句:现在让我们来一条一条分析一下。

String s = "Fred"; 结论:创建了一个String对象

这条语句相当于String s = new String("Fred");
因此,毫无疑问,第一条语句创建了一个String对象,我想没有有疑问吧?

s = s + "47"; 结论:未创建String对象

这条语句也许很多人认为是创建了String对象,我一开始也是这么认为的。但是为了验证我的想法。决定
用点法术恢复这条语句的本来面目。(有很多时候,编译器总是在里面搞一些小动作,javac.exe也不例外)

现在找到这个程序所生成的.class文件(假设是Test.class),找一个反编译工具,我推荐JAD,可以http://www.softpedia.com/progDownload/JAD-Download-85911.html下载
下载后,有一个jad.exe,将其路径放到环境变量path中(只限windows)。并在.class文件的当前路径执行如下的命令:

jad Test

然后大喊一声“还我本来面目”

会在当前目录下生成一个Test.jad文件,打开它,文件内容如下:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
publicStringmakinStrings()
{
Strings
="Fred";
s
=(newStringBuilder(String.valueOf(s))).append("47").toString();
s
=s.substring(2,5);
s
=s.toUpperCase();
returns.toString();
}

哈哈,其他的语句都没变,只有第二条变长了,虽然多了个new,但是建立的是StringBuilder对象。原来
这是java编译器的优化处理。原则是能不建String对象就不建String对象。而是用StringBuilder对象
加这些字符串连接起来,相当于一个字符串队列。这种方式尤其被使用在循环中,大家可以看看下面的代码:
String s = "";
for(int i=0; i < 10000000; i++)
s += "aa";
没有哪位老大认为这是建立了10000000个String对象吧。但不幸的是,上面的代码虽然没有建立10000000个String对象
但却建立了10000000个StringBuilder对象,那是为什么呢,自已用jad工具分析一下吧。
正确的写法应该是:

StringBuilder sb = new StringBuilder("");
for(int i=0; i < 10000000; i++)
sb.append(String.valueOf(i));

s = s.substring(2, 5); 结论:创建了一个String对象
也许有很多人一开始就认为这条语句是创建了一个String对象,那么恭喜你,这条语句确实创建了一个String对象
实际上就是substring方法创建了一个String对象。这也没什么复杂的,自已下一个JDK源代码,看看substring是如何实现的
就可以知道了。我先说一下吧。先不用管substring是如何实现的,反正在substring方法返回时使用了一个new显式地建立了一个String对象
不信自己看看源码。
s = s.toUpperCase(); 结论:创建了一个String对象

toUpperCase()和substring方法类似,在返回时也是使用了new建立了一个String对象。

return s.toString(); 结论:未创建String对象

toString方法返回的就是this,因此,它的返回值就是s。

这道题还算比较简单,再给大家出一个更复杂一点的,也是关于String对象的创建的(只是改了一个原题)。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->publicStringmakinStrings()
{
Strings
="Fred";
s
=s+"Iloveyou.".substring(1).toLowerCase();
s
=s.substring(0);
s
=s.substring(0,1).toUpperCase();
returns.toString();
}


先公布答案吧,上述代码也创建了3个String对象,哈哈!


为什么呢?

要想知道为什么,先得弄清楚substring、toLowerCase和toUpperCase什么时候创建String对象,什么时候不创建对象。

substring方法在截取的子字符串长度等于原字符串时,直接返回原字符串。并不创建新的String对象。

toLowerCase方法在字符串中更本没有需要转换的大写字母时直接返回原字符串,如"abcd".toLowerCase()直接返回abcd,并不创建新的String对象

toUpperCase方法和toLowerCase类似。"ABCD".toUpperCase()直接返回ABCD。


知道了这个,上面的代码就非常清楚了。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->publicStringmakinStrings()
{
Strings
="Fred";//创建一个String对象
s=s+"Iloveyou.".substring(1).toLowerCase();//substring(1)创建一个String对象,由于toLowerCase()转换的字符串是"loveyou.",没有大写字母,因此,它不创建新的String对象
s=s.substring(0);//由于substring(0)截获的是s本身,因此,这条语句不创建新的String对象
s=s.substring(0,1).toUpperCase();//substring(0,1)创建了一个String对象,但由于substring(0,1)的结果是"F",为一个大写字母,因此,toUpperCase直接返回"F"本身。
returns.toString();
}




国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

分享到:
评论

相关推荐

    JAVA面试题解惑系列(二)——到底创建了几个String对象-JAVA程序员JAVA工程师面试必看.pdf,这是一份不错的文件

    JAVA 面试题解惑系列之 String 对象创建机制 本文将深入探讨 JAVA 中 String 对象的创建机制,解答常见的面试题目,并探索 String 对象池的概念和机制。 一、String 对象的创建方式 在 JAVA 中,String 对象可以...

    Java解惑(中文版)_java_java解惑_solve65p_

    通过《Java解惑》这本书,初学者不仅可以系统地学习Java语言,还能获得解决实际问题的技巧,从而更好地适应编程环境并提升编程能力。书中的65个问题或主题涵盖了上述各个知识点,确保读者在学习过程中能够逐步建立起...

    JAVA面试题解惑系列合集

    字符串是Java中非常特殊的对象类型,字符串的创建和拼接有很多细节,比如字符串连接使用'+'时,会生成多个String对象,而使用StringBuilder或者StringBuffer则更为高效。另外字符串池的存在也使得字符串的操作更加...

    臧圩人--JAVA面试题解惑系列合集.pdf

    **1.2 JAVA面试题解惑系列(二)——到底创建了几个String对象?** - **知识点**:探讨Java中String对象的创建机制,特别是常量池的概念和String对象的内存分配策略。了解字符串字面量与`new String()`的区别,以及...

    Java_解惑(PDF)

    "Java_解惑"这份PDF文档旨在帮助初学者解决他们在学习过程中遇到的常见问题和困惑,提供清晰的解释和指导。以下是一些可能涵盖的知识点: 1. **基础语法**:Java的基础语法包括数据类型(如整型、浮点型、字符型和...

    java解惑(包括pdf和答案)

    "java解惑"这个主题旨在帮助初学者理解和解决在学习Java过程中遇到的问题,通过实例来深入浅出地讲解Java的基础知识,同时也强调了实用技巧和注意事项。 "Java解惑"的资料可能包含了两部分:`.chm`和`.pdf`格式的...

    Java面试题解惑系列

    **知识点概述**:在Java中,如何确定创建了多少个String对象,通常涉及到字符串常量池的概念以及字符串的创建方式。 **核心概念**: 1. **字符串常量池**:Java中有一个特殊的内存区域用来存储字符串字面量,这个...

    JAVA面试题解惑系列

    Java中的String对象是不可变的,因此创建String时需要理解其内存机制。面试中常见的问题是关于字符串连接或复制操作时,实际创建了多少个对象。例如,"abc" + "def" 实际上会产生几个String对象?答案是可能会有两...

    JAVA面试题解惑系列合集.docx

    本文通过对Java面试题中的两个典型问题——类的初始化顺序和String对象的创建——进行了深入分析。这些知识点不仅是Java编程的基础,也是深入理解Java语言特性的关键。掌握这些内容不仅能够帮助开发者在面试中表现...

    JAVA面试题解惑--经典面试题

    (二)关于创建 String 对象的问题,Java中的String是不可变的,这意味着一旦创建了一个String对象,就不能更改它的值。因此,每次对String进行操作,如连接、修改等,都会创建新的String对象。例如,"abc" + "def" ...

    Java解惑PPT6

    在"Java解惑PPT6"中,我们探讨了几个关键的Java特性,特别是关于不变性、equals()和hashCode()方法的约定以及它们在HashSet中的应用。 首先,让我们来看一下Library Puzzle(Puzzle 56)中的Big Problem。这个例子...

Global site tag (gtag.js) - Google Analytics