再谈Java-String

一个案例

有以下案例:

public class Main {
\
    public static void main(String[] args) {
        String s1 = "hello";
        String s2 = "hello";
        String s3 = new String("hello");
        String s4 = new String("hello");
        System.out.println(s1 == s2);
        System.out.println(s1 == s3);
        System.out.println(s3 == s4);
    }

}

输出结果为:

true

false

false

为什么s1和s2是同一个对象为什么呢?

在Java程序中,类似于:1,2,3,3.14,“hello"等字面类型的常量经常频繁使用,为了使程序的运行速度更快更节省内存,Java为8种基本数据类型和String类都提供了常量池。

这个字符串常量池底层就是一个哈希表, 名为StringTable. 也就是一个String对应一个Value.

什么是池?

池是编程中经常有的概念, 是一种重要的提升效率的方式, 我们会经常遇到各种池, 例如线程池, 内存池等等.

为了节省空间, 和提高运行效率, java中引用了:

  1. Class文件常量池

  1. 运行时常量池

  1. 字符串常量池

字符串常量池

这个字符串常量池底层使用c/c++写的, 我们在jvm中去看源码的话,会发现其实这个常量池是一个StringTable的类, 实际上是一个固定大小的哈希表

String对象的创建

一个完整的String对象:

可以看到里面有一个字符数字, 然后还有一个hash值, 其中这个hash值为0.

图解

那么对于字符串常量池, 字符串的存储又是怎么样的呢?

直接使用字符串常量赋值

public class Main {
    public static void main(String[] args) {
        String s1 = "hello";
        String s2 = "hello";
        System.out.println( s1 == s2); // true
    }
}

运行结果为True;

首先String s1是在main方法中的, 理论上他是一个局部变量,是一个字符串的引用, 他是被放在栈区的.

首先创建字符串S1, 将其放入常量池, 然后创建s2, 然后遍历哈希表, 发现哈希表对应的哈希值已经存在, 所以直接将这个s2引用指向s1的那个字符串.

只要是new的对象,都是唯一的

intern方法

intern方法是一个本地(native)方法, 底层使用c++实现, 看不到其实现的源代码, 该方法的作用是手动将创建的String对象, 添加到常量池当中去.

一个例子:

public class Main {
    public static void main(String[] args) {
        char[] ch = new char[] {'a','b','c'};
        String stringCh = new String(ch);
        String s1 = "abc";
        System.out.println(s1 == stringCh);
    }
}

提问: 在创建了stringCh对象只有, 常量池当中有abc字符串吗?

看运行结果:

结果为false, 说明stringCh中的"abc"没有被放入常量池, 此时再和s1比较的话, 就是两个不同的对象.

如果使用intern将其加入到常量字符串呢?

public class Main {
    public static void main(String[] args) {
        char[] ch = new char[] {'a','b','c'};
        String stringCh = new String(ch);
        stringCh.interm();
        String s1 = "abc";
        System.out.println(s1 == stringCh);
    }
}

结果为true, 可以看到intern方法把stringCh放入了常量池.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值