【119期】面试官问,深入理解Nginx及使用Nginx实现负载均衡

System.out.println(System.identityHashCode(str3));

System.out.println(System.identityHashCode(str4));

}

}

得到下面一组打印输出,同一个对象的hash值肯定是相同的,而下面str1和str2的hash值不同肯定不是同一个对象(注意不能调用String重写的hashCode(),我们要调用Object提供的native修饰的hashCode()或者利用System.identityHashCode()得到hash值(这种情况就相当于是内存地址))

false    // str1和str2不是同一个对象

460141958     // 堆空间创建的那个对象 ”java“

1163157884   // 字符串常量池中的字符串对象

true    // 说明str3和str4是同一个对象

1956725890   // 堆中创建的字符串对象 ”helloworld“

1956725890   // 堆中创建的字符串对象 ”helloworld“

先讲一讲前面2行的由来

helloWorld

true

我们都知道String类有一个intern()方法,它的作用就是将字符串存入常量池中,并且方法执行完后将这个字符串对象返回。更多面试题推荐公众号Java精选,回复Java面试,获取最新最全面试题,支持在线随时随地刷题。

不难理解第一次打印前在常量池中没有helloWorld字符串,因此会将这个对象存入常量池中。然后返回字符串打印了第一个helloWorld字符串。在比较==的 intern()方法返回的是常量池中的字符串对象(也是前面创建的对象,两个对象是同一个对象),所以返回了第一个true

为了证明,我们可以通过javap -v TestDemo.class命令将字节码文件反编译得到如下字节码。

下面请阅读一遍:

字节码中分常量池、方法test01是我们重点关注的地方,首先注意到常量池中已经有了hello、World、ja、va字符串。

原因在于我们代码中使用String str1 = new StringBuilder("hello").append("World").toString();这里面"hello"这类就是常量,接着我们直接读test01方法

public class com.example.demo.test.TestDemo

minor version: 0

major version: 58

flags: (0x0021) ACC_PUBLIC, ACC_SUPER

this_class: #47 // com/example/demo/test/TestDemo

super_class: #2 // java/lang/Object

interfaces: 0, fields: 0, methods: 2, attributes: 1

Constant pool:

#1 = Methodref #2.#3 // java/lang/Object.“”😦)V

#2 = Class #4 // java/lang/Object

#3 = NameAndType #5:#6 // “”😦)V

#4 = Utf8 java/lang/Object

#5 = Utf8

#6 = Utf8 ()V

#7 = Class #8 // java/lang/StringBuilder

#8 = Utf8 java/lang/StringBuilder

#9 = String #10 // 常量"hello"字符串

#10 = Utf8 hello

#11 = Methodref #7.#12 // java/lang/StringBuilder.“”:(Ljava/lang/String;)V

#12 = NameAndType #5:#13 // “”:(Ljava/lang/String;)V

#13 = Utf8 (Ljava/lang/String;)V

#14 = String #15 // 常量"World"字符串

#15 = Utf8 World

#16 = Methodref #7.#17 // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;

#17 = NameAndType #18:#19 // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;

#18 = Utf8 append

#19 = Utf8 (Ljava/lang/String;)Ljava/lang/StringBuilder;

#20 = Methodref #7.#21 // java/lang/StringBuilder.toString:()Ljava/lang/String;

#21 = NameAndType #22:#23 // toString:()Ljava/lang/String;

#22 = Utf8 toString

#23 = Utf8 ()Ljava/lang/String;

#24 = Fieldref #25.#26 // java/lang/System.out:Ljava/io/PrintStream;

#25 = Class #27 // java/lang/System

#26 = NameAndType #28:#29 // out:Ljava/io/PrintStream;

#27 = Utf8 java/lang/System

#28 = Utf8 out

#29 = Utf8 Ljava/io/PrintStream;

#30 = Methodref #31.#32 // java/lang/String.intern:()Ljava/lang/String;

#31 = Class #33 // java/lang/String

#32 = NameAndType #34:#23 // intern:()Ljava/lang/String;

#33 = Utf8 java/lang/String

#34 = Utf8 intern

#35 = Methodref #36.#37 // java/io/PrintStream.println:(Ljava/lang/String;)V

#36 = Class #38 // java/io/PrintStream

#37 = NameAndType #39:#13 // println:(Ljava/lang/String;)V

#38 = Utf8 java/io/PrintStream

#39 = Utf8 println

#40 = Methodref #36.#41 // java/io/PrintStream.println:(Z)V

#41 = NameAndType #39:#42 // println:(Z)V

#42 = Utf8 (Z)V

#43 = String #44 // 常量"ja"字符串

#44 = Utf8 ja

#45 = String #46 // 常量"va"字符串

#46 = Utf8 va

#47 = Class #48 // com/example/demo/test/TestDemo

#48 = Utf8 com/example/demo/test/TestDemo

#49 = Utf8 Code

#50 = Utf8 LineNumberTable

#51 = Utf8 LocalVariableTable

#52 = Utf8 this

#53 = Utf8 Lcom/example/demo/test/TestDemo;

#54 = Utf8 test01

#55 = Utf8 str1

#56 = Utf8 Ljava/lang/String;

#57 = Utf8 str2

#58 = Utf8 str3

#59 = Utf8 StackMapTable

#60 = Utf8 RuntimeVisibleAnnotations

#61 = Utf8 Lorg/junit/Test;

#62 = Utf8 SourceFile

#63 = Utf8 TestDemo.java

{

public com.example.demo.test.TestDemo();

descriptor: ()V

flags: (0x0001) ACC_PUBLIC

Code:

stack=1, locals=1, args_size=1

0: aload_0

1: invokespecial #1 // Method java/lang/Object.“”😦)V

4: return

LineNumberTable:

line 7: 0

LocalVariableTable:

Start Length Slot Name Signature

0 5 0 this Lcom/example/demo/test/TestDemo;

public void test01();

descriptor: ()V

flags: (0x0001) ACC_PUBLIC

Code:

stack=3, locals=4, args_size=1

0: new #7 // 创建 java/lang/StringBuilder 对象

3: dup

4: ldc #9 // 入栈常量池(#9)中的字符串常量hello

6: invokespecial #11 // 实例初始化将hello传入

9: ldc #14 // 入栈常量池(在#14)中的字符串常量World

11: invokevirtual #16 // StringBuilder.append方法调用传入World

14: invokevirtual #20 // 调用StringBuilder.toString方法

17: astore_1 // 将返回的值的地址引用存入到局部变量1

18: getstatic #24 // 获取打印流

21: aload_1 // 将局部变量1装载成引用类型,也就是"helloWorld"

22: invokevirtual #30 // "helloWorld"调用String.intern:()方法

25: invokevirtual #35 // 调用打印流,打印"helloWorld"

28: getstatic #24 // 获取打印流

31: aload_1 // 加载局部变量1 也就是"helloWorld"

32: aload_1 // 加载局部变量1 “helloWorld”

33: invokevirtual #30 // "helloWorld"调用intern方法

36: if_acmpne 43 // 如果条件满足就转执行43 的iconst_0也就是将0入栈

39: iconst_1 // 将int类型常量值1压入栈

40: goto 44 // 无条件转移到44

43: iconst_0 // 将0入栈

44: invokevirtual #40 // 调用方法传入也就是“ja"(对应常量池的#44)

47: new #7 // 创建StringBuilder对象

50: dup

51: ldc #43 // 加载字符串"ja"

53: invokespecial #11 // 实例化StringBuilder

56: ldc #45 // 加载字符串"va"

58: invokevirtual #16 // 调用append方法

61: invokevirtual #20 // 调用toString方法

64: astore_2 // 将toString的结果存入局部变量2中

65: getstatic #24 // 获取打印流

68: aload_2 // 加载局部变量2的值也就是加载"java"

69: invokevirtual #30 // 调用intern方法

72: invokevirtual #35 // 打印"java"

75: getstatic #24 // 获取打印流

78: aload_2 // 加载局部变量2"java"

79: aload_2 // 加载局部变量2"java"

80: invokevirtual #30 // 调用intern方法

83: if_acmpne 90 // 如果条件成立跳转到90也就是将0入栈

86: iconst_1 // 将常量值1入栈

87: goto 91 // 无条件跳转到91

90: iconst_0 // 将常量值0入栈

91: invokevirtual #40 // 打印

94: new #7 // class java/lang/StringBuilder

97: dup

98: ldc #9 // String hello

100: invokespecial #11 // Method java/lang/StringBuilder.“”:(Ljava/lang/String;)V

103: invokevirtual #20 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;

106: astore_3

107: getstatic #24 // Field java/lang/System.out:Ljava/io/PrintStream;

110: aload_3

111: invokevirtual #30 // Method java/lang/String.intern:()Ljava/lang/String;

114: invokevirtual #35 // Method java/io/PrintStream.println:(Ljava/lang/String;)V

117: getstatic #24 // Field java/lang/System.out:Ljava/io/PrintStream;

120: aload_3

小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
img

而且极易碰到天花板技术停滞不前!**

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-OUjw6PhP-1710900486317)]
[外链图片转存中…(img-YTI4eim1-1710900486318)]
[外链图片转存中…(img-ME0Q1LYe-1710900486319)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
[外链图片转存中…(img-S86QgwTu-1710900486319)]

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值