Java中String的intern()方法詳細(xì)說(shuō)明
String的intern()方法
先看第一個(gè)例子
public class HelloWorld {
public static void main(String[] args) {
String s1 = new String("hello") + new String("world");
s1.intern();
String s2 = "helloworld";
System.out.println(s1==s2);
}
}
輸出為:true
再看第二個(gè)例子
public class HelloWorld {
public static void main(String[] args) {
String s1 = new String("hello") + new String("world");
String s2 = "helloworld";
s1.intern();
System.out.println(s1==s2);
}
}
輸出為:false
明明只有一行代碼的位置發(fā)生了改變,為什么會(huì)輸出截然不同的結(jié)果呢?
下面開始分析。
jdk1.7后的intern方法在調(diào)用后,存在兩種情況。
一、在調(diào)用后發(fā)現(xiàn)StringTable中沒(méi)有所對(duì)應(yīng)的字符串
String s1 = new String("hello") + new String("world");
這句代碼會(huì)在堆中new一個(gè)String對(duì)象(其實(shí)并沒(méi)有這么簡(jiǎn)單,后面講)
s1.intern();
s1調(diào)用了intern方法,然后發(fā)現(xiàn)在StringTable中并沒(méi)有所對(duì)應(yīng)的字符串,那么jvm就會(huì)在StringTable中放入一個(gè)地址,這個(gè)地址指向s1所new的String對(duì)象。
String s2 = "helloworld";
jvm發(fā)現(xiàn)在StringTable中已經(jīng)存在指向"helloworld"的地址了,就會(huì)把StringTable中的地址給s2。此時(shí)s1和s2都指向這個(gè)地址。所以是同一個(gè)對(duì)象。于是返回了true
二、在調(diào)用后發(fā)現(xiàn)StringTable中存在所對(duì)應(yīng)的字符串
String s1 = new String("hello") + new String("world");
這句代碼會(huì)在堆中new一個(gè)String對(duì)象(其實(shí)并沒(méi)有這么簡(jiǎn)單,后面講)
String s2 = "helloworld";
jvm在StringTable中存儲(chǔ)這一個(gè)"helloworld"。
此時(shí)s1和s2是兩個(gè)不同的對(duì)象。s1指向堆中,s2指向StringTable。
s1.intern();
jvm發(fā)現(xiàn)在StringTable中已經(jīng)存在"helloworld"了,那么什么都不會(huì)做,僅僅是返回這一個(gè)字符串。所以結(jié)果為false
下面講一下String s1 = new String("hello") + new String("world"); 底層做了什么。
public class HelloWorld {
public static void main(String[] args) {
String s1 = new String("hello") + new String("world");
}
}
先運(yùn)行一遍,類加載器加載后,在target中會(huì)存在HelloWorld.class
public class HelloWorld {
public HelloWorld() {
}
public static void main(String[] args) {
String s1 = new String("hello") + new String("world");
}
}
運(yùn)行指令javap -v target/classes/aa/HelloWorld.class進(jìn)行反編譯。 反編譯后的詳細(xì)信息如下:
Classfile /C:/Users/top/Downloads/untitled5/target/classes/aa/HelloWorld.class
Last modified 2022年4月14日; size 677 bytes
SHA-256 checksum bf071098fdd9e51e927c011bb021cab16ac1227bd18a6435441f30e30b400f70
Compiled from "HelloWorld.java"
public class aa.HelloWorld
minor version: 0
major version: 52
flags: (0x0021) ACC_PUBLIC, ACC_SUPER
this_class: #10 // aa/HelloWorld
super_class: #11 // java/lang/Object
interfaces: 0, fields: 0, methods: 2, attributes: 1
Constant pool:
#1 = Methodref #11.#27 // java/lang/Object."<init>":()V
#2 = Class #28 // java/lang/StringBuilder
#3 = Methodref #2.#27 // java/lang/StringBuilder."<init>":()V
#4 = Class #29 // java/lang/String
#5 = String #30 // hello
#6 = Methodref #4.#31 // java/lang/String."<init>":(Ljava/lang/String;)V
#7 = Methodref #2.#32 // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
#8 = String #33 // world
#9 = Methodref #2.#34 // java/lang/StringBuilder.toString:()Ljava/lang/String;
#10 = Class #35 // aa/HelloWorld
#11 = Class #36 // java/lang/Object
#12 = Utf8 <init>
#13 = Utf8 ()V
#14 = Utf8 Code
#15 = Utf8 LineNumberTable
#16 = Utf8 LocalVariableTable
#17 = Utf8 this
#18 = Utf8 Laa/HelloWorld;
#19 = Utf8 main
#20 = Utf8 ([Ljava/lang/String;)V
#21 = Utf8 args
#22 = Utf8 [Ljava/lang/String;
#23 = Utf8 s1
#24 = Utf8 Ljava/lang/String;
#25 = Utf8 SourceFile
#26 = Utf8 HelloWorld.java
#27 = NameAndType #12:#13 // "<init>":()V
#28 = Utf8 java/lang/StringBuilder
#29 = Utf8 java/lang/String
#30 = Utf8 hello
#31 = NameAndType #12:#37 // "<init>":(Ljava/lang/String;)V
#32 = NameAndType #38:#39 // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
#33 = Utf8 world
#34 = NameAndType #40:#41 // toString:()Ljava/lang/String;
#35 = Utf8 aa/HelloWorld
#36 = Utf8 java/lang/Object
#37 = Utf8 (Ljava/lang/String;)V
#38 = Utf8 append
#39 = Utf8 (Ljava/lang/String;)Ljava/lang/StringBuilder;
#40 = Utf8 toString
#41 = Utf8 ()Ljava/lang/String;
{
public aa.HelloWorld();
descriptor: ()V
flags: (0x0001) ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
LineNumberTable:
line 3: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Laa/HelloWorld;
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: (0x0009) ACC_PUBLIC, ACC_STATIC
Code:
stack=4, locals=2, args_size=1
0: new #2 // class java/lang/StringBuilder
3: dup
4: invokespecial #3 // Method java/lang/StringBuilder."<init>":()V
7: new #4 // class java/lang/String
10: dup
11: ldc #5 // String hello
13: invokespecial #6 // Method java/lang/String."<init>":(Ljava/lang/String;)V
16: invokevirtual #7 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
19: new #4 // class java/lang/String
22: dup
23: ldc #8 // String world
25: invokespecial #6 // Method java/lang/String."<init>":(Ljava/lang/String;)V
28: invokevirtual #7 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
31: invokevirtual #9 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
34: astore_1
35: return
LineNumberTable:
line 5: 0
line 6: 35
LocalVariableTable:
Start Length Slot Name Signature
0 36 0 args [Ljava/lang/String;
35 1 1 s1 Ljava/lang/String;
}
SourceFile: "HelloWorld.java"

可以看到,
1、jvm先new了一個(gè)StringBuilder(非多線程安全),
2、然后new了一個(gè)String,
3、調(diào)用了StringBuilder的append方法,添加了這一個(gè)String對(duì)象。
4、再次new一個(gè)String對(duì)象
5、再次調(diào)用了StringBuilder的append方法,添加了這一個(gè)String對(duì)象。
6、調(diào)用StringBuilder的toString方法,將StringBuilder轉(zhuǎn)為String
到此這篇關(guān)于Java中String的intern()方法詳細(xì)說(shuō)明的文章就介紹到這了,更多相關(guān)String的intern()方法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Boot條件注解之@ConditionalOnProperty完全解析
這篇文章主要介紹了SpringBoot中的@ConditionalOnProperty注解,通過(guò)配置文件屬性值控制Bean或配置類的加載,實(shí)現(xiàn)功能開關(guān)和環(huán)境配置,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2025-02-02
Spring Boot應(yīng)用開發(fā)初探與實(shí)例講解
這篇文章主要介紹了Spring Boot應(yīng)用開發(fā)初探與實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
關(guān)于Spring?Validation數(shù)據(jù)校檢的使用流程分析
在實(shí)際項(xiàng)目中,對(duì)客戶端傳遞到服務(wù)端的參數(shù)進(jìn)行校驗(yàn)至關(guān)重要,SpringValidation提供了一種便捷的方式來(lái)實(shí)現(xiàn)這一需求,通過(guò)在POJO類的屬性上添加檢查注解,本文給大家介紹Spring?Validation數(shù)據(jù)校檢的使用流程,感興趣的朋友一起看看吧2024-11-11
Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(52)
下面小編就為大家?guī)?lái)一篇Java基礎(chǔ)的幾道練習(xí)題(分享)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧,希望可以幫到你2021-08-08
MyBatis關(guān)閉一級(jí)緩存的兩種方式(分注解和xml兩種方式)
這篇文章主要介紹了MyBatis關(guān)閉一級(jí)緩存的兩種方式(分注解和xml兩種方式),mybatis默認(rèn)開啟一級(jí)緩存,執(zhí)行2次相同sql,但是第一次查詢sql結(jié)果會(huì)加工處理這個(gè)時(shí)候需要關(guān)閉一級(jí)緩存,本文給大家詳細(xì)講解需要的朋友可以參考下2022-11-11
springboot + rabbitmq 如何實(shí)現(xiàn)消息確認(rèn)機(jī)制(踩坑經(jīng)驗(yàn))
這篇文章主要介紹了springboot + rabbitmq 如何實(shí)現(xiàn)消息確認(rèn)機(jī)制,本文給大家分享小編實(shí)際開發(fā)中的一點(diǎn)踩坑經(jīng)驗(yàn),內(nèi)容簡(jiǎn)單易懂,需要的朋友可以參考下2020-07-07
logback EvaluatorFilter實(shí)現(xiàn)同時(shí)記錄多個(gè)level級(jí)別的日志
這篇文章主要介紹了logback EvaluatorFilter實(shí)現(xiàn)同時(shí)記錄多個(gè)level級(jí)別的日志方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
SpringBoot事件監(jiān)聽器@EventListener的實(shí)現(xiàn)
@EventListener注解用于處理應(yīng)用程序事件,提供了一種方便的方式來(lái)監(jiān)聽和響應(yīng)各種事件,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-11-11

