串池StringTable与s.intern()方法 、 验证s.intern()方法(基于jdk1.8)(附1.6对比)

本文详细比较了Java 1.8和1.6中`s.intern()`方法的行为,着重讲解了字符串常量池的使用和不同版本对对象处理方式的变更。通过实例展示了在字符串拼接后调用`intern()`前后的不同结果。
摘要由CSDN通过智能技术生成

验证s.intern()方法(基于jdk1.8)(附1.6对比)

1、在 jdk1.8中 s.intern()方法指的是尝试将字符串对象s放入串池中 :如果串池中有,则不会放入;若没有,放入(和堆中为同一对象)。
  • 最终返回的是串池中的对象(不论放不放入)
  • 如:String s1 = s.intern(); s1指的是串池中的常量对象
2、下面是两个测试横向对比(1.8环境):
两个方法不同之处:
  • 主要在执行String s1 = s.intern() 前后,具体看下面测试方法的注释
  • 导致两个方法中最后一行System.out.println("ab" == s) 输出不同
测试一:执行String s1 = s.intern() 前放入"ab"
//最终串池中对象StringTable["a", "b", "ab" ]
public static void main(String[] args) {

    //vm调用StringBuild.append("a").append("b").toString() 最终相当于变成 new String("ab")
    String s = new String("a") + new String("b");

    //*****线程此时串池中对象StringTable["a", "b"]

    //堆内存: new String("a")   new String("b")   new String("ab")

    System.out.println("ab" == s);  //false  串池中没有"ab",此次会将 "ab"常量放入串池 因为 s在堆内存中,"ab"在串池中内存地址不同

    //****此时串池中对象StringTable["a", "b", "ab"]

    //s.intern尝试将字符串对象s放入串池中 :如果串池中有,则不会放入;若没有,放入, 最终会返回串池中的对象
    String s1 = s.intern();  //不会放入s,因为串池中已经存在 "ab"

    System.out.println("ab" == s1);  //true   都是串池中的常量对象 ab

    System.out.println("ab" == s);  //false   s为堆内存地址 "ab"为串池中

}
测试二:执行String s1 = s.intern() 后放入"ab"
//最终串池中对象StringTable["a", "b", "ab" ]
@Test
public void test() {
    //vm调用StringBuild.append("a").append("b").toString() 最终相当于变成 new String("ab")
    String s = new String("a") + new String("b");  //简化等同于  String s = new String("ab");

    //*****线程此时串池中对象StringTable["a", "b"]

    //堆内存: new String("a")   new String("b")   new String("ab")

    //s.intern尝试将字符串对象s放入串池中 :如果串池中有,则不会放入;若没有,放入, 最终会返回串池中的对象
    String s1 = s.intern();  //放入s,因为串池中不存在 "ab" 之前代码行中未出现常量 "ab"

    System.out.println("ab" == s1);  //true   都是串池中的常量对象 ab

    //****此时串池中对象StringTable["a", "b", "ab"]

    System.out.println("ab" == s);   //true   s、"ab"为串池中同一个常量
}
3、附:在 jdk1.6中 s.intern()方法指的是尝试将字符串对象s放入串池中 :如果串池中有,则不会放入;若没有,放入时为拷贝一份s对象放入
测试:
//最终串池中对象StringTable["a", "b", "ab" ]
public static void main(String[] args) {

    String s = new String("a") + new String("b");

    //*****线程此时串池中对象StringTable["a", "b"]

    String s1 = s.intern();  //拷贝一份s对象放入

    //*****线程此时串池中对象StringTable["a", "b", "ab"]
    
    System.out.println("ab" == s1);  //true   都是串池中的常量对象 ab

    System.out.println("ab" == s);   //false   s为堆内存地址 "ab"为串池中对象

}
# 总结:
  • 1.6将常量对象放入常量池时拷贝,最终堆中和常量池中对象不同
  • 1.8将常量对象放入常量池时直接将堆内存中对象放入,最终堆中和常量池中对象为同一个

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值