一尘不染

在Java中使用String(String)构造函数

java

我读过一些文章和书籍,String s = newString("...");几乎总是避免使用它。我知道为什么会这样,但是使用String(String)构造函数有什么用吗?我不认为有,也没有其他证据,但是我想知道SO社区中的任何人是否知道使用。


阅读 571

收藏
2020-09-08

共1个答案

一尘不染

这是一篇很好的文章:认为无用的String构造函数毕竟是有用的!

事实证明,在至少一种情况下,此构造函数实际上可以有用。如果您曾经看过String源代码,就会发现它不仅包含用于char数组值和字符数的字段,而且还包含用于String开头的偏移量的字段。这样一来,字符串就可以与其他字符串共享char数组值,通常是通过调用substring()方法之一导致的。早在几年前,jwz的Java咆哮就为此对Java进行了严厉的谴责:

造成这种开销的唯一原因是String.substring()可以返回共享相同值数组的字符串。这样做的代价是向每个String对象添加8个字节并不是净节省…

除了字节节省,如果您有以下代码:

// imagine a multi-megabyte string here  
String s = "0123456789012345678901234567890123456789";  
String s2 = s.substring(0, 1);  
s = null;

现在,您将拥有一个String
s2,尽管它似乎是一个字符的字符串,但它包含对在String中创建的巨大char数组的引用。这意味着即使我们显式使String无效,该数组也不会被垃圾回收!

解决此问题的方法是使用我们前面提到的“无用的” String构造函数,如下所示:

String s2 = new String(s.substring(0, 1));

众所周知,如果旧数组大于字符串中的字符数,则此构造函数实际上会将旧内容复制到新数组中。这意味着旧的String内容将按预期进行垃圾回收。快乐快乐快乐快乐。

最后,Kat Marsen提出了以下几点:

首先,字符串常量永远不会被垃圾回收。第二,字符串常量是内部变量,这意味着它们将在整个VM中共享。这样可以节省内存。但这并不总是您想要的。

String上的复制构造函数允许您从String文字创建私有String实例。这对于构造有意义的互斥对象(出于同步目的)非常有价值。

2020-09-08