蚂蚁面试:字符串在JVM中如何存放?
字符串對象在JVM中可能有兩個存放的位置:字符串常量池或堆內存。
-
使用常量字符串初始化的字符串對象,它的值存放在字符串常量池中;
-
使用字符串構造方法創建的字符串對象,它的值存放在堆內存中;
String提供了一個API,?java.lang.String.intern(),這個API可以手動將一個字符串對象的值轉移到字符串常量池中。
在1.7之前,字符串常量池是在PermGen區域,這個區域的大小是固定的——不能在運行時根據需要擴大,也不能被垃圾收集器回收,因此如果程序中有太多的字符串調用了intern方法的話,就可能造成OOM。
在1.7以后,字符串常量池移到了堆內存中,并且可以被垃圾收集器回收,這個改動降低了字符串常量池OOM的風險。
案例分析
驗證代碼:
public class StringTest {public static void main(String[] args) {String s1 = "javaadu";String s2 = "javaadu";String s3 = new String("javaadu");System.out.println(s1 == s2); //trueSystem.out.println(s1 == s3); //falseString s4 = s3.intern();System.out.println(s1 == s4); //true}}intern源碼分析
intern方法的實現底層是一個native方法,在Hotspot JVM里字符串常量池它的邏輯在注釋里寫得很清楚:如果常量池中有這個字符串常量,就直接返回,否則將該字符串對象的值存入常量池,再返回。
這里以Openjdk1.8的源碼為例,跟下intern方法的底層實現,String.java文件對應的C文件是String.c:
JNIEXPORT jobject JNICALLJava_java_lang_String_intern(JNIEnv *env, jobject this) {return JVM_InternString(env, this); }JVM_InternString這個方法的定義在jvm.h,實現在jvm.cpp中,在JVM中,Java世界和C++世界的連接層就是jvm.h和jvm.cpp這兩文件。
JVM_ENTRY(jstring, JVM_InternString(JNIEnv *env, jstring str))JVMWrapper("JVM_InternString");JvmtiVMObjectAllocEventCollector oam;if (str == NULL) return NULL;oop string = JNIHandles::resolve_non_null(str);oop result = StringTable::intern(string, CHECK_NULL);return (jstring) JNIHandles::make_local(env, result);JVM_END可以看出,字符串常量池在JVM內部就是一個HashTable,也就是上面代碼中的StringTable。
從 StringTable::intern方法跟下去,就可以發現:如果找到了這次操作的字符串,就直接返回found_string;如果沒有找到,就將當前的字符串加入到HashTable中,然后再返回。
總結
在Java應用恰當得使用String.intern()方法有助于節省內存空間,但是在使用的時候,也需要注意,因為StringTable的大小是固定的,如果常量池中的字符串過多,會影響程序運行效率。
總結
以上是生活随笔為你收集整理的蚂蚁面试:字符串在JVM中如何存放?的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 这个开源项目帮你将Linux命令行一网打
- 下一篇: 20万用户同时访问一个热点Key,如何优