JAVA基础知识指导:Java中String三种加法的区别
i! Q, t0 j0 Y: {. ^" H* d8 [4 G J- K( w: i( F+ Y0 c$ ~
java它提供了如此丰富的library,让程序员能够很容易的写出功能强大的程序。同时它也封装了如此多的细节,让程序员能够轻易的写出很拙劣的程序。它所代表的object指向思想设计之初是为了把程序员从纷繁复杂的编程细节中解放出来,来达到对已有程序的利用。但是在编程实际上又要求程序员了解其实现的细节,而避免写出因调用已有library不正引起的performance慢的、耗费系统资源多的程序。通常成为一个优秀的Java程序员需要较长时间的经验积累,包括从程序的tuning中或从其他有经验的程序员口中,才知道一定功能需要怎样实现,在程序中需要避免那些问题。但这往往是比较片面的,知其然而不知其所以然。我想大多数程序员都有类似的经历,举一个简单的例子,下面有三种对String的累加操作。
- A3 D# H' C+ ?* n2 \
) P( |" O1 W* L# w ①String tmp = "a" + "b" + "c";
. w4 h+ N5 ?- u - z2 ~8 w0 D4 Z" R
②String tmp = null;5 A f# v1 p# a! C- Y% {
( A9 w+ ~' Y3 b# v8 L0 N, P
tmp+= "a";
. A- x1 v+ K1 m) |0 [
( U) P# N8 r: D1 J. W' v tmp+= "b";
+ P; U: Y1 O3 G / K% L. `6 I, W* J u5 g, {# j' }
tmp+= "c";
* o1 {7 V; g/ z; S2 ?
* `) \; ^. {% v1 J3 m1 t& q( P ③String tmp = null;
+ d+ V. q7 A5 w! G( q
8 T9 I0 z; @) K* W: _ StringBuffer buf = new StringBuffer();: q# {( i: v0 D/ w3 w, s, e
' N- a2 W# y3 Z8 m G3 U C
buf.append("a");
. ]$ [) h) Q' m - u8 R* ]5 D& z% L& r
buf.append("b");( ?* [0 F# a7 p. Z4 A' q
& Z u5 k4 d8 f* x5 [9 \8 {: c
buf.append("c");
) {6 j8 r) ]& j+ g. n ) U9 U: \) ?& m5 |8 R9 l% Z
tmp = buf.toString();: t/ `8 p& \0 Q
* Z5 H: p3 F% z9 f8 n5 J* Z
有些Java程序员在任意的用这三种方法的任一种,无视它们的区别。有些程序员知道第三种方法好,一直在用而不知其为什么好,以至于作为经验教条的传授给 Java新手。真正的答案是什么呢?让我们揭开Java String Class和StringBuffer Class的封装面纱,看看它的内部实现。在Java中的String Class是一个不可变类,所有对一个Sting Object的改变都会导致一个新的String Object的生成。那么对tmp+=a中+符号的实现呢?
2 m' m% i/ k$ n' C6 i) B 5 s: C% t2 M l$ P1 W4 Q! e# K
如果你注意一下StringBuffer的Javadoc会发现,JDK对它的实现是:tmp=(new StringBuffer()。append(tmp)。append("a"))。toSting();这样我们发现②这种方法在隐性生成了一个 StringBuffer Object和一个String Object 再乘3就是六个object 的资源耗费。(还不包括String Class和StringBuffer Class内部使用的Char[])。而方法③只用了两个。也许有些程序员会对这些耗费不以为然,的确也是,不过在一些场景下它会成为 perfamence的瓶颈。再回头看看①这种方法,它会被Java编译器编译为:tmp=(new StringBuffer()。append("a")。append("b")。append("c"))。to String();我们发现做的和③方法是同样的事。 |