阿里面试题:字符串对象在JVM中是如何存放的

Posted 阿飞的博客

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了阿里面试题:字符串对象在JVM中是如何存放的相关的知识,希望对你有一定的参考价值。

典型答案


字符串对象在JVM中可能有两个存放的位置:字符串常量池或堆内存。

  • 使用常量字符串初始化的字符串对象,它的值存放在字符串常量池中;

  • 使用字符串构造方法创建的字符串对象,它的值存放在堆内存中;

String提供了一个API——java.lang.String.intern(),这个API可以手动将一个字符串对象的值转移到字符串常量池中。

在1.7之前,字符串常量池是在PermGen区域,这个区域的大小是固定的——不能在运行时根据需要扩大,也不能被垃圾收集器回收,因此如果程序中有太多的字符串调用了intern方法的话,就可能造成OOM。

在1.7以后,字符串常量池移到了堆内存中,并且可以被垃圾收集器回收,这个改动降低了字符串常量池OOM的风险。


知识点总结


2
案例分析




验证代码:

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); //true
        System.out.println(s1 == s3); //false

        String s4 = s3.intern();
        System.out.println(s1 == s4); //true
    }
}
2
intern源码分析

intern方法的实现底层是一个native方法,在Hotspot JVM里字符串常量池它的逻辑在注释里写得很清楚:如果常量池中有这个字符串常量,就直接返回,否则将该字符串对象的值存入常量池,再返回。

这里以Openjdk1.8的源码为例,跟下intern方法的底层实现,String.java文件对应的C文件是String.c:

JNIEXPORT jobject JNICALL
Java_java_lang_String_intern(JNIEnv *envjobject this)
{
   return JVM_InternString(envthis);
}

JVM_InternString这个方法的定义在jvm.h,实现在jvm.cpp中,在JVM中,Java世界和C++世界的连接层就是jvm.h和jvm.cpp这两文件。

JVM_ENTRY(jstringJVM_InternString(JNIEnv *envjstring str))
 JVMWrapper("JVM_InternString");
 JvmtiVMObjectAllocEventCollector oam;
 if (str == NULLreturn NULL;
 oop string = JNIHandles::resolve_non_null(str);
 oop result = StringTable::intern(stringCHECK_NULL);
 return (jstringJNIHandles::make_local(envresult);
JVM_END

可以看出,字符串常量池在JVM内部就是一个HashTable,也就是上面代码中的StringTable。

StringTable::intern方法跟下去,就可以发现:如果找到了这次操作的字符串,就直接返回found_string;如果没有找到,就将当前的字符串加入到HashTable中,然后再返回。

总结


在Java应用恰当得使用String.intern()方法有助于节省内存空间,但是在使用的时候,也需要非常注意,因为StringTable的大小是固定的,如果常量池中的字符串过多,那么就会导致每个entry上的元素过多,从而影响程序效率,导致YGC的STW时间变长(可以通过参数-XX:StringTableSize=2500000缓解这个问题,但是不能彻底解决,可参考笔者另一片文章:)。

↓↓↓↓


以上是关于阿里面试题:字符串对象在JVM中是如何存放的的主要内容,如果未能解决你的问题,请参考以下文章

阿里面试官:字符串在JVM中如何存放?90%的人就真的只回答在哪里存放

016:字符串对象在JVM中是如何存放的

蚂蚁面试:字符串在JVM中如何存放?

常问面试:String 对象在 JVM 如何存储提高性能的?

10道精选阿里+华为Java面试题JVM+多线程+Redis+Mysql(附答案)

Java中JVM相关面试题-整理