String 为什么用 final 修饰?
在进入 class 学习后,我们不仅会接触到 String,还会接触到包装类。什么是包装类呢?String 和 包装类都有一个共同点。
1、包装类
Java 中的基本数据类型没有方法和属性,而包装类就是为了让这些拥有方法和属性,实现对象化交互。
基本数据类型
包装类
2、包装类和 String 类为什么都用 final 修饰?
① 源码
String
Integer
② 为什么 String 要设计成 final?
我们都知道 String 类的不可变型。用 final 修饰意味着不可继承和修改。当一个 String 变量被第二次赋值时,不是在原有内存地址上修改数据,而是在内存中重新开辟一块内存地址,并指向新地址。
设计成 final 的好处:
(1)不可变性支持线程安全
(2)不可变性支持字符串常量池,提升性能
(3)String 经常在 Map 中作为 key 值,保证 HashCode 的不可变性。
你是否弄懂了 String?小试牛刀一下
图1
输出结果:
见评论区,测测你和其他人的答案是否一样
聊聊图1中的情况 3
采用 new 关键字新建一个字符串对象时,JVM 首先在字符串池中查找有没有 "abc" 这个字符串对象,如果有,则不在池中再去创建 "abc" 这个对象了,直接在堆中创建一个 "abc" 字符串对象,然后将堆中的这个 "abc" 对象的地址返回赋给引用 s5,这样,s5 就指向了堆中创建的这个 "abc" 字符串对象;如果没有,则首先在字符串池中创建一个 "abc" 字符串对象,然后再在堆中创建一个 "abc" 字符串对象,然后将堆中这个 "abc" 字符串对象的地址返回赋给 s5 引用,这样,s5指向了堆中创建的这个 "abc" 字符串对象。s6 则指向了堆中创建的另一个 "abc" 字符串对象。s5 、s6 是两个指向不同对象的引用,结果当然是 false。
如果你觉得很绕,说明你 Java 基础这块有待提高。
情况 4
虽然 s7 和 s8 各自指的是常量池里面 "a","b" 的引用,但是 String 在做加法或者 subString、replace 等方法的时候,实际上返回的是 new String () 的结果,因此 str 指向的是堆中的地址,所以不相等。
情况 6
虽然此处 ss 是 new 出来的新对象,但是由于调用了 intern 方法,这个方法会返回常量池中相等值的字符串的地址,所以最后 ss 指向的是常量池中 "abc" 的地址,所以相等。
String、StringBuffer 和 StringBuild
1、String 不可变字符序列,如果要操作少量的数据用 String;
2、StringBuffer 可变字符序列、效率低和线程安全,多线程操作字符串缓冲区下操作大量数据用 StringBuffer
StringBuffer 中的方法大都采用了 synchronized 关键字进行修饰
3、StringBuild 可变字符序列,效率高和线程不安全,单线程操作字符串缓冲区下操作大量数据用StringBuilder
为什么说 StringBuild 是线程不安全?
1、测试
输出结果:999
很明显,输出的结果和我们预期的不一致。
如果这个时候,面试官问你,StringBuild 什么地方会导致线程不安全?
① 我们定位到 appen 方法
我们看第 450 行, count += len ,这不是原子操作
如果多线程下,这个 count 值 必然出现问题
@Python大星 | 文
如果你还遇到其他的 String 相关的面试题,欢迎分享评论区,咱一起盘盘。