[每天一个知识点]13-String对象与常量的老生常谈

有种很常见的面试题是关于何种情况下会创建对象何种情况下不会创建对象的,对于字符串常见的就是

String str1 = "abc";
String str2 = new String("abc");
String str3 = String.valueOf("abc");

等等情况。其实我并没有很想清楚对新人的面试考查这些细节问题的目的是什么,其实只要记住一条,绝不轻易new String对象和原生类型的包装类型就可以了,至于这里面的细节差异,绝不是新人能搞懂以及需要搞懂的。

下面还是简单讲讲上面几行代码,下面的描述其实并非规格的一部分,只能代表某一个实现,不过基本原理差不多是一致的。

第1行代码后堆里有一个String对象,至于是不是在这行代码执行后新创建的,要看字符串池里之前有没有这个字符串,至于有没有,后面详细说。在java里面,创建一个对象的概念你可以认为是在堆里分配一段空间、做初始化工作(分配id、各种元数据等)、调用构造方法,这些都做完才叫创建一个对象,在java里对象一定是有id的。

第1行代码在执行时做了这样一个操作:查看字符串池里面有没有"abc"这一项,如果有直接给str1赋值对应的对象,如果没有就创建一个对象,放到池里同时也赋值给str1。从这个流程也就可以明白,String str1 = "abc";到底有没有创建对象是不一定的。同时需要注意的是,java7之后的字符串池也是可以被垃圾回收的,所以未必再次执行String str = "abc";就一定不会再创建对象了。

第2行代码一定会去创建一个新对象,然后用"abc"这个对象去初始化,所以在某些情况下这一行代码会创建2个对象。其实扩展到任何类型都是成立的,只要是new一定会有新对象。对于String和原生类型的包装类型除非你非常确定自己一定需要一个新对象,否则就用=或者valueOf。

第3行代码执行时会绑定到

public static String valueOf(Object obj) {
    return (obj == null) ? "null" : obj.toString();
}

上,然后再看String.toString()的实现就明白了,它只有1行代码

public String toString() {
    return this;
}

稍微提一点进一步的信息,就是String的intern()方法,它其实针对新产生的对象,它会触发一次往字符串池里面扔的操作。但是这个操作很微妙也没有什么显式调用的必要,不去关心它最好。如果面试中级软件工程师及以下的岗位被问到本篇的问题以及intern(),我敢保证面试官也十有八九不完全了解它。

最后强调作为最佳实践,对于String和各种包装类型来说,优先使用=字面值的方式(String str = "abc"; Integer i = 1;),其次是valueOf,如果非常清楚自己确实需要一个新对象,才使用new。


你可能感兴趣的:(每天一个知识点)