• String.intern() 和常量池


    4.1 String 类和常量池

    String 对象的两种创建方式:

    String str1 = "abcd";//先检查字符串常量池中有没有"abcd",如果字符串常量池中没有,则创建一个,然后 str1 指向字符串常量池中的对象,如果有,则直接将 str1 指向"abcd"";
    String str2 = new String("abcd");//堆中创建一个新的对象
    String str3 = new String("abcd");//堆中创建一个新的对象
    System.out.println(str1==str2);//false,一个是堆上对象,一个是字符串常量池中对象
    System.out.println(str2==str3);//false,每次new都生成一个对象,两个对象地址不同,所以false

    这两种不同的创建方法是有差别的。

    • 第一种双引号的方式是在常量池中创建对象;
    • 第二种 new 方式是直接在堆内存空间创建一个新的对象。

    记住一点:只要使用 new 方法,便需要创建新的对象。

    再给大家一个图应该更容易理解,图片来源:https://www.journaldev.com/797/what-is-java-string-pool

    String-Pool-Java

    String 类型的常量池对象的创建方式有两种:

    • 直接使用双引号声明出来的 String 对象会直接存储在常量池中。
    • 如果不是用双引号声明的 String 对象,可以使用 String 提供的 intern 方法。String.intern() 是一个 Native 方法,它的作用是:如果运行时常量池中已经包含一个等于此 String 对象内容的字符串,则返回常量池中该字符串的引用;如果没有,JDK1.7之前(不包含1.7)的处理方式是在常量池中创建与此 String 内容相同的字符串,并返回常量池中创建的字符串的引用,JDK1.7以及之后的处理方式是在常量池中记录此字符串的引用,并返回该引用。
     String s1 = new String("计算机");//堆上对象
     String s2 = s1.intern(); //常量池中对象
     String s3 = "计算机";//常量池对象,因为“计算机”已存在于常量池,所以直接返回常量池引用
     System.out.println(s2);//计算机
     System.out.println(s1 == s2);//false,因为一个是堆内存中的 String 对象一个是常量池中的 String 对象,
     System.out.println(s3 == s2);//true,因为两个都是常量池中的 String 对象
    

    字符串拼接: String + String 是堆上对象,"" + "" 是常量池对象

    String str1 = "str";
    String str2 = "ing";//两者都是常量池对象
    		 
    String str3 = "str" + "ing";//常量池中的对象
    String str4 = str1 + str2; //在堆上创建的新的对象	  
    String str5 = "string";//常量池中的对象
    System.out.println(str3 == str4);//false
    System.out.println(str3 == str5);//true
    System.out.println(str4 == str5);//false
    

      字符串拼接

    尽量避免多个字符串拼接,因为这样会重新创建对象。如果需要改变字符串的话,可以使用 StringBuilder 或者 StringBuffer。

    4.2 String s1 = new String("abc");这句话创建了几个字符串对象?

    将创建 1 或 2 个字符串。如果运行时常量池中已存在字符串常量“abc”,则只会在堆空间创建一个字符串常量“abc”。如果池中没有字符串常量“abc”,那么它将首先在池中创建,然后在堆空间中创建,因此将创建总共 2 个字符串对象。

    验证:

    String s1 = new String("abc");// 堆内存的地址值
    String s2 = "abc";//常量池地址
    System.out.println(s1 == s2);// 输出 false,因为一个是堆内存,一个是常量池的内存,故两者是不同的。
    System.out.println(s1.equals(s2));// 值相同,输出 true
  • 相关阅读:
    第4章 排序
    第5章 算术与代数
    第6章 组合数学
    第7章 数论
    第8章 回溯法
    第9章 图遍历
    第11章 动态规划
    第10章 图算法
    第12章 网格
    第13章 几何
  • 原文地址:https://www.cnblogs.com/jjfan0327/p/12789524.html
Copyright © 2020-2023  润新知