Java中String的intern()方法詳細說明
String的intern()方法
先看第一個例子
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
再看第二個例子
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ā)生了改變,為什么會輸出截然不同的結(jié)果呢?
下面開始分析。
jdk1.7后的intern方法在調(diào)用后,存在兩種情況。
一、在調(diào)用后發(fā)現(xiàn)StringTable中沒有所對應的字符串
String s1 = new String("hello") + new String("world");
這句代碼會在堆中new一個String對象(其實并沒有這么簡單,后面講)
s1.intern();
s1調(diào)用了intern方法,然后發(fā)現(xiàn)在StringTable中并沒有所對應的字符串,那么jvm就會在StringTable中放入一個地址,這個地址指向s1所new的String對象。
String s2 = "helloworld";
jvm發(fā)現(xiàn)在StringTable中已經(jīng)存在指向"helloworld"的地址了,就會把StringTable中的地址給s2。此時s1和s2都指向這個地址。所以是同一個對象。于是返回了true
二、在調(diào)用后發(fā)現(xiàn)StringTable中存在所對應的字符串
String s1 = new String("hello") + new String("world");
這句代碼會在堆中new一個String對象(其實并沒有這么簡單,后面講)
String s2 = "helloworld";
jvm在StringTable中存儲這一個"helloworld"。
此時s1和s2是兩個不同的對象。s1指向堆中,s2指向StringTable。
s1.intern();
jvm發(fā)現(xiàn)在StringTable中已經(jīng)存在"helloworld"了,那么什么都不會做,僅僅是返回這一個字符串。所以結(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"); } }
先運行一遍,類加載器加載后,在target中會存在HelloWorld.class
public class HelloWorld { public HelloWorld() { } public static void main(String[] args) { String s1 = new String("hello") + new String("world"); } }
運行指令javap -v target/classes/aa/HelloWorld.class進行反編譯。 反編譯后的詳細信息如下:
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了一個StringBuilder(非多線程安全),
2、然后new了一個String,
3、調(diào)用了StringBuilder的append方法,添加了這一個String對象。
4、再次new一個String對象
5、再次調(diào)用了StringBuilder的append方法,添加了這一個String對象。
6、調(diào)用StringBuilder的toString方法,將StringBuilder轉(zhuǎn)為String
到此這篇關于Java中String的intern()方法詳細說明的文章就介紹到這了,更多相關String的intern()方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
SpringBoot如何訪問不同的數(shù)據(jù)庫的方法實現(xiàn)
本文主要介紹了在SpringBoot應用中配置和管理多個數(shù)據(jù)源的方法,包括使用SpringBoot官方支持的配置方式和第三方庫實現(xiàn)多數(shù)據(jù)源配置,感興趣的可以了解一下2024-11-11基于Properties類操作.properties配置文件方法總結(jié)
這篇文章主要介紹了Properties類操作.properties配置文件方法總結(jié),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09Java貪心算法之Prime算法原理與實現(xiàn)方法詳解
這篇文章主要介紹了Java貪心算法之Prime算法原理與實現(xiàn)方法,簡單描述了Prime算法的概念、原理、實現(xiàn)與使用技巧,需要的朋友可以參考下2017-09-09Netty分布式pipeline管道傳播事件的邏輯總結(jié)分析
這篇文章主要為大家介紹了Netty分布式pipeline管道傳播事件總結(jié)分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-03-03