【Java面试题】请说出下面程序的输出

Java String相关

来源: 黑马程序员的《Java面试宝典(Beta6.0)》二五6. 请说出下面程序的输出(2017-11-14-wl)

一、原内容

请说出下面程序的输出(2017-11-14-wl)

    class StringEqualTest {
        public static void main(String[] args) {
            String s1 = "Programming";
            String s2 = new String("Programming");
            String s3 = "Program";
            String s4 = "ming";
            String s5 = "Program" + "ming";
            String s6 = s3 + s4;
            System.out.println(s1 == s2); //false
            System.out.println(s1 == s5); //true
            System.out.println(s1 == s6); //false
            System.out.println(s1 == s6.intern()); //true
            System.out.println(s2 == s2.intern()); //false
        }
    }

补充:解答上面的面试题需要知道如下两个知识点:

  1. String 对象的 intern()方法会得到字符串对象在常量池中对应的版本的引用(如果常量池中有一个字符串与String 对象的 equals 结果是 true),如果常量池中没有对应的字符串,则该字符串将被添加到常量池中,然后返回常量池中字符串的引用;
  2. 字符串的+操作其本质是创建了 StringBuilder 对象进行 append 操作,然后将拼接后的 StringBuilder 对象用 toString 方法处理成 String 对象,这一点可以用 javap -c StringEqualTest.class 命令获得 class 文件对应的 JVM 字节码指令就可以看出来。

二、看完原文解释还是一脸懵,so上网搜去。

1、网络摘抄

以下内容来源(摘自页面中下面的笔记):
https://www.runoob.com/java/java-string-intern.html
(看完以下内容,上面的输出结果就很清晰了)

尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。

可以看下面一个范例:

    String str1 = "a";
    String str2 = "b";
    String str3 = "ab";
    String str4 = str1 + str2;
    String str5 = new String("ab");
     
    System.out.println(str5.equals(str3)); //true
    System.out.println(str5 == str3); //false
    System.out.println(str5.intern() == str3); //true
    System.out.println(str5.intern() == str4); //false

为什么会得到这样的一个结果呢?我们一步一步的分析。

  • 第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。
  • 第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false
  • 第三、当str5调用intern()的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。
  • 第四,当str4 = str1 + str2后,str4的值也为”ab”,但是为什么这个结果会是false呢?先看下面代码:
    String a = new String("ab");
    String b = new String("ab");
    String c = "ab";
    String d = "a" + "b";
    String e = "b";
    String f = "a" + e;
    
    System.out.println(b.intern() == a); //false
    System.out.println(b.intern() == c); //true
    System.out.println(b.intern() == d); //true
    System.out.println(b.intern() == f); //false
    System.out.println(b.intern() == a.intern()); //true

由运行结果可以看出来,b.intern() == ab.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == db.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。

当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal==的区别,当时记得老师只是说==判断的是“地址”,但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。

执行下面的代码:

    String a = "abc";
    String b = "abc";
    String c = "a" + "b" + "c";
    String d = "a" + "bc";
    String e = "ab" + "c";
            
    System.out.println(a == b); //true
    System.out.println(a == c); //true
    System.out.println(a == d); //true
    System.out.println(a == e); //true
    System.out.println(c == d); //true
    System.out.println(c == e); //true

2、关键点总结

  • ==表示对内存地址进行比较,equals()表示对字符串的内容进行比较。
  • 采用new创建的字符串对象不进入字符串池。
  • 字符串相加的时候,都是静态字符串的结果会添加到字符串池;如果其中含有变量不会添加字符串池中。
  • 在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用
  • 9
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值