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

先看看下面的代码

    public String makinStrings()
jA)Dc%v(Ah;I:j ]0     {ITPUB个人空间'pB ^m8~8K�N
        String s = "Fred";ITPUB个人空间'z_}j(c]
        s = s + "47";ITPUB个人空间CR1l_b&E
        s = s.substring(2, 5);ITPUB个人空间#`%I+e;i$c F1Nc
        s = s.toUpperCase();
9q c;u"`/X{,l)FF0         return s.toString();ITPUB个人空间+M-FM;o ^ Z4W9R~.n
    }

 

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

 

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

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

这条语句相当于String s = new String("Fred");
+_9I![aM;K&pi}|0 因此,毫无疑问,第一条语句创建了 一个String对象,我想没有有疑问吧?

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

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

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

jad Test

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

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

... ...ITPUB个人空间~ oo1|P�T[
    public String makinStrings()ITPUB个人空间 MHyO-W:g Y
    {ITPUB个人空间r)BfC*K#p
        String s = "Fred";
U)naI4K P_0         s = (new StringBuilder(String.valueOf(s))).append("47").toString();ITPUB个人空间"M ]�| `7o%bi
        s = s.substring(2, 5);ITPUB个人空间3BI1Stx E
        s = s.toUpperCase();ITPUB 个人空间s+v/Eq(E
        return s.toString();
'l�iB7f ^0     }
KsQ@/'[(c;e`0 ... ...

哈哈,其他的语句都没变,只有第二条变长了,虽然多了个new,但是建立的是StringBuilder对象。原来
3V*jmUst0 这是java 编 译器的优化处理。原则是能不建String对象就不建String对象。而是用StringBuilder对象ITPUB个人空间H5y||)Ls!Id
加这些字符串连接起来,相当 于一个字符串队列。这种方式尤其被使用在循环中,大家可以看看下面的代码:
%jA(Ty q0         String s = "";
Pm4Y a7f?0         for(int i=0; i < 10000000; i++)ITPUB个人空 间tJ�T|Zn
            s += "aa";
.K(z,cH:il0 没有哪位老大认为这是建立了 10000000个String对象吧。但不幸的是,上面的代码虽然没有建立10000000个String对象
~!h"i.j x9cn`:y2}0 但却建立了10000000个StringBuilder对象,那是为什么呢,自已用jad工具分析一下吧。ITPUB个人空间 },WJuT+x
正确的写法应该是:

        StringBuilder sb = new StringBuilder("");ITPUB个人空间9Xh@N4g
        for(int i=0; i < 10000000; i++)
.A W9d3B0o3P ms*o0             sb.append(String.valueOf(i));

 s = s.substring(2, 5);     结论:创建了一个String对象ITPUB个人空间EE!?xD S F(Q
 也 许有很多人一开始就认为这条语句是创建了一个String对象,那么恭喜你,这条语句确实创建了一个String对象ITPUB个人空间Q aK1x"B~?K*H
 实际上就是 substring方法创建了一个String对象。这也没什么复杂的,自已下一个JDK源代码,看看substring是如何实现的ITPUB个人空间Q_]*t7zm.|v/V/^8i-A
 就可以知 道了。我先说一下吧。先不用管substring是如何实现的,反正在substring方法返回时使用了一个new显式地建立了一个String对象
/W,Dgp�u[0  不信自己看看源码。ITPUB个人空间fXds /
s = s.toUpperCase();   结论:创建了一个String对象

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

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

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

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

    public String makinStrings()ITPUB个人空间Y?6^K9Eb"oV/
    {
I0E-usY:}!U0         String s = "Fred"; ITPUB个人空间:};u h,Stj
        s = s + "Iloveyou.".substring(1).toLowerCase();
Vlg3?7V ?3Ei o0         s = s.substring(0);ITPUB个人空间U8P(Q;[IZ;[c
        s = s.substring(0,1).toUpperCase();
$EW$`|K'w&Aa0         return s.toString();
%F!l)X(lTP0M@ H0     }

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

 

为什么呢?

 

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

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

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

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

 

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

 

    public String makinStrings()
#w#a'| Yu D}0     {ITPUB个人空间xW(`7O7a(S:u y/V"?
        String s = "Fred";     //创建一个String对象 ITPUB 个人空间2L U@S"F�Fzc
        s = s + "Iloveyou.".substring(1).toLowerCase();  //substring(1)创建一个 String对象 ,由于toLowerCase()转换的字符串是"loveyou.",没有大写字母,因此,它不创建新的 String对象ITPUB个人空间1~!` Qv rt)E c$rr
        s = s.substring(0);   // 由于substring(0)截获的是s本身,因此,这条语句不创建新的String对象ITPUB个人空间V:ts-Y(D,ba)E
        s = s.substring(0,1).toUpperCase();  //substring(0,1)创建了一个String对象 , 但由于substring(0,1)的结果是"F",为一个大写字母,因此,toUpperCase直接返回"F"本身。
$E:^d*{L/I#cb{7s0         return s.toString();ITPUB个人空间u ^"K;a-O
    }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值