Java基础 -- String 类和常量池String s1 = new String(“abc”)这句话创建了几个字符串对象8 种基本类型的包装类和常量池

Posted CodeJiao

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java基础 -- String 类和常量池String s1 = new String(“abc”)这句话创建了几个字符串对象8 种基本类型的包装类和常量池相关的知识,希望对你有一定的参考价值。

文章目录

1. String类和常量池

        String str1 = "abcd";//先检查字符串常量池中有没有"abcd",如果字符串常量池中没有,则创建一个,然后 str1 指向字符串常量池中的对象,如果有,则直接将 str1 指向"abcd""
        String str2 = "abcd";
        String str3 = new String("abcd");//堆中创建一个新的对象
        String str4 = new String("abcd");//堆中创建一个新的对象
        System.out.println(str1 == str2);// true
        System.out.println(str1 == "abcd");// true
        System.out.println(str1 == str3);//false
        System.out.println(str3 == str4);//false

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

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

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


2. String 类型的常量池的两种主要使用方法

  1. 直接使用双引号声明出来的 String 对象会直接存储在常量池中。
  2. 如果不是用双引号声明的 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 对象

运行结果:


2.1 字符串拼接

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

可以参考下面篇文章:

Java基础 – 字符串类(String、StringBuffer、StringBuilder)、自动装箱与拆箱

        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

运行结果:


2. String s1 = new String(“abc”)这句话创建了几个字符串对象

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

验证:

        String s1 = new String("abc");// 堆内存的地址值
        String s2 = "abc";
        // 输出 false,因为一个是堆内存, 一个是常量池的内存, 故两者是不同的。
        System.out.println(s1 == s2);
        // 输出 true, 内存地址不同, 但是字符串的值是相同的. 说明有相同值的字符串在2个不同的地方创建(堆 / 常量池)
        System.out.println(s1.equals(s2));

结果:


3. 8 种基本类型的包装类和常量池

  • Java 基本类型的包装类的大部分都实现了常量池技术,即 ByteShortIntegerLongCharacterBoolean;这 5 种包装类默认创建了数值[-128,127] 的相应类型的缓存数据(性能和资源之间的权衡),但是超出此范围仍然会去创建新的对象。
  • 两种浮点数类型的包装类 FloatDouble 并没有实现常量池技术。

Integer 缓存源代码:

应用场景:

  • Integer i1=40;Java 在编译的时候会直接将代码封装成 Integer i1=Integer.valueOf(40);,从而使用常量池中的对象。
  • Integer i1 = new Integer(40);这种情况下会创建新的对象。

        Integer i1 = 40;
        Integer i2 = 40;
        Integer i3 = 0;
        Integer i4 = new Integer(40);
        Integer i5 = new Integer(40);
        Integer i6 = new Integer(0);
        System.out.println("i1=i2 " + (i1 == i2));//true
        System.out.println("i1=i2+i3 " + (i1 == i2 + i3));//true
        System.out.println("i1=i4 " + (i1 == i4));// false
        System.out.println("i4=i5 " + (i4 == i5));// false
        System.out.println("i4=i5+i6 " + (i4 == i5 + i6));//true
        System.out.println("40=i5+i6 " + (40 == i5 + i6));//true

解释:

语句 i4 == i5 + i6,因为+这个操作符不适用于 Integer 对象,首先 i5i6 进行自动拆箱操作,进行数值相加,即 i4 == 40。然后 Integer 对象无法与数值进行直接比较,所以 i4 自动拆箱转为 int40,最终这条语句转为 40 == 40 进行数值比较。



以上是关于Java基础 -- String 类和常量池String s1 = new String(“abc”)这句话创建了几个字符串对象8 种基本类型的包装类和常量池的主要内容,如果未能解决你的问题,请参考以下文章

JAVA String介绍常量池及StringStringBuilder和StringBuffer得区别. 以及8种基本类型的包装类和常量池得简单介绍

jvm虚拟机

Java基础字符串与数组

java 面试基础

在java中String的传参

对String的内存解析