Java面试题之最扯淡的String

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java面试题之最扯淡的String相关的知识,希望对你有一定的参考价值。


SB里面的toString方法如下,为什么需要这个,查看JVM虚拟机指令用,+号会变成new SB()的,然后调用toString方法
  public String toString() {
        // Create a copy, don‘t share the array
        return new String(value, 0, count);
    }

  


public class ThreadException { public static void main(String[] args) {
        没加final的代码 String hello = "hello"; String hel = "hel"; String lo = "lo"; System.out.println(hello == "hel" + "lo"); System.out.println(hello == "hel" + lo); } }
没加final的代码反编译回来后的代码
import java.io.PrintStream;

public class ThreadException
{
public static void main(String[] paramArrayOfString)
{
String str1 = "hello";
String str2 = "hel";
String str3 = "lo";
System.out.println(str1 == "hello");
System.out.println(str1 == "hel" + str3);
}
}
没加final的虚拟机执行的指令集如下
public static void main(java.lang.String[]);
Code:
0: ldc #2 // String hello
2: astore_1
3: ldc #3 // String hel
5: astore_2
6: ldc #4 // String lo
8: astore_3   //将操作数栈顶的值保存在本地变量表,变量地址放本地变量表3位置  
9: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
12: aload_1
13: ldc #2 // String hello
15: if_acmpne 22
18: iconst_1
19: goto 23
22: iconst_0
23: invokevirtual #6 // Method java/io/PrintStream.println:(Z)V
26: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
29: aload_1
30: new #7 // class java/lang/StringBuilder
33: dup
34: invokespecial #8 // Method java/lang/StringBuilder."<init>":()V
37: ldc #3 // String hel    //这句加载常量到操作数栈  hel
39: invokevirtual #9 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
42: aload_3          //加载一个本地变量到操作数栈,从本地变量表3这个位置,将里面的值放到操作数栈上去
43: invokevirtual #9 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
46: invokevirtual #10 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;//SB.tostring
49: if_acmpne 56
52: iconst_1
53: goto 57
56: iconst_0
57: invokevirtual #6 // Method java/io/PrintStream.println:(Z)V
60: return

  再看下面这个,多加了一个final

加了final的代码
public class ThreadException { public static void main(String[] args) { String hello = "hello"; String hel = "hel"; final String lo = "lo"; System.out.println(hello == "hel" + "lo");//true System.out.println(hello == "hel" + lo);//true } }
加了final后反编译回来的代码
import java.io.PrintStream; public class ThreadException { public static void main(String[] paramArrayOfString) { String str4 = "hello"; String str5 = "hel"; System.out.println(str4 == "hello"); System.out.println(str4 == "hello"); } }

看下加了final的虚拟机代码 
public static void main(java.lang.String[]);
Code:
0: ldc #2 // String hello   //加载常量到操作数栈,常量是hello
2: astore_1
3: ldc #3 // String hel
5: astore_2
6: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
9: aload_1
10: ldc #2 // String hello
12: if_acmpne 19
15: iconst_1
16: goto 20
19: iconst_0
20: invokevirtual #5 // Method java/io/PrintStream.println:(Z)V
23: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;
26: aload_1
27: ldc #2 // String hello  加载常量
29: if_acmpne 36    比较
32: iconst_1
33: goto 37
36: iconst_0
37: invokevirtual #5 // Method java/io/PrintStream.println:(Z)V
40: return
}

  再来看这段代码

public class ThreadException
{

  加了final public static void main(String[] args) { String s = new String("abc"); final String s1 = "abc"; String s2 = new String("abc"); final String s3 = "abc"; System.out.println(s == s.intern()); System.out.println(s1 == s2.intern()); System.out.println(s2.intern() == s2.intern()); System.out.println(s3 == s1); } }
加了final反编译回来的代码

import java.io.PrintStream;

public class ThreadException
{
  public static void main(String[] paramArrayOfString)
  {
    String str1 = new String("abc");

    String str2 = new String("abc");

    System.out.println(str1 == str1.intern());
    System.out.println("abc" == str2.intern());
    System.out.println(str2.intern() == str2.intern());
//这段是不是很惊讶
    System.out.println(true);
  }
}
是不是很扯淡?

  再看没加final的

public class ThreadException
{

	public static void main(String[] args)
	{
  没加final的代码
		String s = new String("abc");
		 String s1 = "abc";
		String s2 = new String("abc");
		 String s3 = "abc";
		System.out.println(s == s.intern());
		System.out.println(s1 == s2.intern());
		System.out.println(s2.intern() == s2.intern());
		System.out.println(s3 == s1);
	}
}
没加final反编译后的代码


import java.io.PrintStream;

public class ThreadException
{
  public static void main(String[] paramArrayOfString)
  {
    String str1 = new String("abc");
    String str2 = "abc";
    String str3 = new String("abc");
    String str4 = "abc";
    System.out.println(str1 == str1.intern());
    System.out.println(str2 == str3.intern());
    System.out.println(str3.intern() == str3.intern());
    System.out.println(str4 == str2);
  }
}

  是不是都感觉很扯淡,还是C++ Primer里面那句话,对于字面量字符串的处理,有些编译器会保存一个,有些会保存多个副本,所以,

     想判断String字面量是否相等,老老实实的for循环

以上是关于Java面试题之最扯淡的String的主要内容,如果未能解决你的问题,请参考以下文章

java面试题之----String的intern

好程序员Java学习路线分享Java面试题之加载机制

java基础面试题之:switch的参数类型

Java面试题之Integer.valueOf(String s);采用了什么设计模式

java面试题之Thread的run()和start()方法有什么区别

java初级面试题之byte b = 130;有没有问题的问题