缓冲池、String/StringBuffer/StringBuilder、intern()

1.缓冲池

1.1 new Integer(123) 与 Integer.valueOf(123) 的区别在于:

  • new Integer(123) :每次都会新建一个对象
  • Integer.valueOf(123) :会使用缓存池中的对象,多次调用会取得同一个对象的引用。
Integer x = new Integer(123);
Integer y = new Integer(123);
System.out.println(x == y); // false
Integer z = Integer.valueOf(123);
Integer k = Integer.valueOf(123);
System.out.println(z == k); // true

1.2 valueOf() 方法的实现

先判断值是否在缓存池中,如果在的话就直接返回缓存池的内容。

public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}

1.3 在 Java 8 中,Integer 缓存池的大小默认为 -128~127。

1.4 自动装箱

编译器会在自动装箱过程调用 valueOf() 方法,因此多个 Integer 实例使用自动装箱来创建并且值相同,那么就会引用相同的对象。

Integer m = 123;
Integer n = 123;
System.out.println(m == n); // true

1.5 基本类型对应的缓冲池如下:

  • boolean values true and false
  • all byte values
  • short values between -128 and 127
  • int values between -128 and 127
  • char in the range \u0000 to \u007F

在使用这些基本类型对应的包装类型时,就可以直接使用缓冲池中的对象

2. String

2.1 概览

String 被声明为 final,因此它不可被继承

内部使用 char 数组存储数据,该数组被声明为 final,这意味着 value 数组初始化之后就不能再引用其它数组。

并且 String 内部没有改变 value 数组的方法,因此可以保证 String 不可变。

public final class String implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];
    ...

2.2 不可变的好处

2.2.1 可以缓存 hash 值

因为 String 的 hash 值经常被使用,例如 String 用做 HashMap 的 key。

不可变的特性可以使得 hash 值也不可变,因此只需要进行一次计算

2.2.2 String Pool 的需要

如果一个 String 对象已经被创建过了,那么就会从 String Pool 中取得引用

只有String 是不可变的,才可能使用 String Pool。

2.2.3 安全性

String 经常作为参数,String 不可变性可以保证参数不可变

例如在作为网络连接参数的情况下如果 String 是可变的,那么在网络连接过程中,String 被改变,改String 对象的那一方以为现在连接的是其它主机,而实际情况却不一定是。

2.2.4 线程安全

String 不可变性天生具备线程安全,可以在多个线程中安全地使用。

3. String, StringBuffer and StringBuilder(详细介绍

3.1. 可变性

  • String 不可变;
  • StringBuffer 和 StringBuilder 可变。

3.2 线程安全

  • String 不可变,因此是线程安全的;
  • StringBuilder 不是线程安全的;
  • StringBuffer 是线程安全的,内部使用 synchronized 进行同步。

3.3 String.intern()

使用 String.intern() 可以保证相同内容的字符串变量引用同一的内存对象

下面示例中,s1 和 s2 采用 new String() 的方式新建了两个不同对象,而 s3 是通过 s1.intern() 方法取得一个对象引用。intern() 首先把 s1 引用的对象放到 String Pool(字符串常量池) 中,然后返回这个对象引用。因此 s3 和 s1 引用的是同一个字符串常量池的对象。

String s1 = new String("aaa");
String s2 = new String("aaa");
System.out.println(s1 == s2); // false
String s3 = s1.intern();
System.out.println(s1.intern() == s3); // true
System.out.println(s1 == s3); // false

如果是采用 "bbb" 这种使用双引号的形式创建字符串实例,会自动地将新建的对象放入 String Pool 中。

String s4 = "bbb";
String s5 = "bbb";
System.out.println(s4 == s5); // true

在 Java 7 之前,字符串常量池被放在运行时常量池中,它属于永久代

而在 Java7,字符串常量池被移到 Native Method 中。这是因为永久代的空间有限,在大量使用字符串的场景下会导致OutOfMemoryError 错误。


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值