• 字符串常量池和String.intern()方法在jdk1.6、1.7、1.8中的变化


    字符串常量池也是运行时常量池

    jdk1.6中,它是在方法区中,属于“永久代”

    jdk1.7中,它被移除方法区,放在java堆中

    jdk1.8中,取消了“永久代”,将常量池放在元空间,与堆独立了

    public class RuntimeConstantPoolOOM {
        public static void main(String[] args) throws Exception{
            String str1 = new StringBuffer("计算机").append("shuju").toString();
            String str2 = new StringBuffer("计算机").append("shuju").toString();
            //jdk1.6的测试效果
            System.out.println(str1.intern()==str1);//false
            System.out.println(str2==str2.intern());//false
            
            //jdk1.7的测试效果
            System.out.println(str1.intern()==str1);//true
            System.out.println(str2==str2.intern());//false
            
            //jdk1.8的测试效果
            System.out.println(str1.intern()==str1);//true
            System.out.println(str2==str2.intern());//false
        }
    }

    String.intern()在JDK1.6中,会先判断常量池中是否存在当前字符串,不存在就会将当前字符串复制到常量池,并返回常量池中字符串的引用。

    而JDK1.7以后,会先判断常量池中是否存在当前字符串,不存在时不会将当前字符串复制到常量池,只是当前字符串的引用。如果存在,也不会改变存在的引用,以及只会保存首次存储的字符串的引用。如上代码所示,常量池保存的是str1的引用,即使str2调用intern()方法,返回的也是str1的引用。

    从以上代码也可以看出,调用toString方法只会在堆中创建字符串对象。

    字符串创建各种状态如下:

    1.只在常量池上创建常量
        String a1 = "AA";
    2.只在堆上创建对象:调用toString方法或者+
        String a2 = new String("A") + new String("A");
    3.在堆上创建对象,在常量池上创建常量
        String a3 = new String("AA");
    4.在堆上创建对象,在常量池上创建引用
        String a4 = new String("A") + new String("A");//只在堆上创建对象AA
        a4.intern();//将该对象AA的引用保存到常量池上

  • 相关阅读:
    深蓝说区块学习笔记
    Golang语言练习
    WebAssembly学习
    JMeter如何维持登录Session状态
    MySQL脏读、不可重复读、幻读及MVCC
    webrtc源码分析(7)-fec
    webrtc源码分析(9)-拥塞控制(下)-码率分配
    webrtc源码分析(8)-拥塞控制(上)-码率预估
    剑指offer刷题合集
    Visual Studio ------- 将在解决方案中单击文件名,预览文件内容功能开启与关闭
  • 原文地址:https://www.cnblogs.com/cherish010/p/10266729.html
Copyright © 2020-2023  润新知