Java当中的常量池
本文轉(zhuǎn)載公眾號(hào)? 達(dá)叔與他的朋友們?
Java當(dāng)中的常量池
在Java虛擬機(jī)jvm中,內(nèi)存分布為:虛擬機(jī)堆,程序計(jì)數(shù)器,本地方法棧,虛擬機(jī)棧,方法區(qū)。
程序計(jì)數(shù)器是jvm執(zhí)行程序的流水線,是用來(lái)存放一些指令的,本地方法棧是jvm操作系統(tǒng)方法所使用的棧,而虛擬機(jī)棧是用來(lái)執(zhí)行程序代碼的棧,在方法區(qū)中有類(lèi)變量,類(lèi)信息,方法信息,常量池(符號(hào)的引用,以表的形式存在的),堆是虛擬機(jī)執(zhí)行程序代碼的所用的堆。
常量?是一旦給定了值就無(wú)法改變的量,用final修飾的成員變量為常量。
什么是class文件常量池?
我們知道在class文件中,有類(lèi)的版本信息,字段信息,方法,接口等信息,還有一個(gè)就是常量池,?這個(gè)就是class文件常量池了。
class文件常量池主要用于存放的是什么呢?
存儲(chǔ)的是編譯生成的各種字面量和符號(hào)引用。在計(jì)算機(jī)科學(xué)中,字面量是用于表達(dá)源代碼中固定值的表示法;而符號(hào)引用是一組符號(hào)用來(lái)描述所引用的目標(biāo),可以是任何形式的字面量,只要使用時(shí)能夠無(wú)歧義的定位到目標(biāo)就行。
常量池是以表的形式存在(表是用來(lái)存儲(chǔ)字符串值的,不存儲(chǔ)符號(hào)引用),實(shí)際可以分兩種,一種為靜態(tài)常量池,另一種為運(yùn)行時(shí)常量池,共有11中常量表,常量池的每一個(gè)常量都代表一張表。
常量表
| CONSTANT_Utf8 | 1 | UTF-8編碼的Unicode字符串 |
| CONSTANT_Integer | 3 | int類(lèi)型的字面值 |
| CONSTANT_Float | 4 | float類(lèi)型的字面值 |
| CONSTANT_Long | 5 | long類(lèi)型的字面值 |
| CONSTANT_Double | 6 | double類(lèi)型的字面值 |
| CONSTANT_Class | 7 | 對(duì)一個(gè)類(lèi)或者是接口的符號(hào)引用 |
| CONSTANT_String | 8 | String類(lèi)型的字面值的引用 |
| CONSTANT_Fieldref | 9 | 對(duì)一個(gè)字段的符號(hào) |
| CONSTANT_Methodref | 10 | 對(duì)一個(gè)類(lèi)中方法的符號(hào)應(yīng)用 |
| CONSTANT_InterfaceMethodref | 11 | 對(duì)一個(gè)接口中方法的符號(hào)引用 |
| CONSTANT_NameAndType | 12 | 對(duì)一個(gè)字段或方法的部分符號(hào)引用 |
常量池
Integer integer1 = 127; Integer integer2 = 127; System.out.println(integer1 == integer2); // trueInteger integer1 = 128; Integer integer2 = 128; System.out.println(integer1 == integer2); // false在Java中符號(hào)“==”是用來(lái)比較地址,符號(hào)“equals”默認(rèn)是與符號(hào)“==”一樣,都是用來(lái)比較地址的。
String string1 = "dashu"; String string2 = "dashu"; System.out.println(string1==string2); // true String string1 = "dashu"; String string3 = new String("dashu"); System.out.println(string1 == string3); // falseString str = new String("dashu");?創(chuàng)建了幾個(gè)對(duì)象呢?
答案是:2個(gè)或者1個(gè)。
在new String("dashu");,如果這個(gè)“dashu”字面值已經(jīng)出現(xiàn)在常量池中,那么就只出創(chuàng)建一個(gè)對(duì)象,如果沒(méi)有就創(chuàng)建兩個(gè)對(duì)象。
原理:?出現(xiàn)了字面量“dashu”,系統(tǒng)會(huì)到字符串常量池中查找是否有相同的字符串存在,如果有,就不會(huì)創(chuàng)建新的對(duì)象了,否則就會(huì)用字面量值“dashu”,創(chuàng)建一個(gè)String對(duì)象。而new String("dashu"),有關(guān)鍵字new的存在,就表示它一定會(huì)創(chuàng)建一個(gè)新的對(duì)象,然后調(diào)用接收String參數(shù)的構(gòu)造器進(jìn)行初始化。
如果改為string1 == string3.intern()結(jié)果為true,因?yàn)榉祷氐氖浅A砍乩锩孀置嬷档牡刂贰?/p>
棧:線程棧和本地方法棧
// 源碼public class Object{ private static native void registerNatives(); static{registerNatives();} } // 源碼public boolean equals(Object obj){ return (this == obj); } // 源碼public String toString(){ return getClass().getName() + "@" + Integer.toHexString(hasCode()); } // 源碼protected native Object clone() throws CloneNotSupportedException;有native修飾符修飾的是通過(guò)JNI來(lái)調(diào)用c語(yǔ)言或是c++執(zhí)行的。
所有的類(lèi)都是Object的子類(lèi)。
萬(wàn)物皆對(duì)象 // 源碼注解Class {@code Object} is the root of the ?class hierarchy. Every class has {@code Object} as a superclass. All objects, including arrays, implements the methods of this class. @ see java.lang Class@ since JDK1.0常量池:
Class文件中存儲(chǔ)所有常量
在Java中說(shuō)過(guò)常量池可以分兩種形態(tài),靜態(tài)常量池和運(yùn)行時(shí)常量池。
靜態(tài)常量池就是class文件中的常量池有字符串字面量,類(lèi)信息,方法的信息等,占用了class文件較大部分的空間,在常量池中主要存放的是字面量和符號(hào)引用量。
運(yùn)行時(shí)常量池是java虛擬機(jī)在完成類(lèi)加載后的操作,將class文件中的常量池加載到內(nèi)存中,并保證在方法區(qū),我們口中的常量池是在方法區(qū)中運(yùn)行的常量池,運(yùn)行時(shí)常量池具有動(dòng)態(tài)性,在運(yùn)行期間也能產(chǎn)生新的常量放入池中,就是上方寫(xiě)過(guò)的代碼。常量不一定要在編譯期間產(chǎn)生,也可以在運(yùn)行期間產(chǎn)生新的產(chǎn)量放入到池中。
如下解析:
Java虛擬機(jī)jvm在執(zhí)行某個(gè)類(lèi)的時(shí)候,要經(jīng)過(guò)類(lèi)從加載到內(nèi)存中,到卸載為止。
整個(gè)過(guò)程為?加載,驗(yàn)證,準(zhǔn)備,解析,初始化,使用,卸載。
-
加載,
-
驗(yàn)證,class文件的版本是否能兼容當(dāng)前的Java虛擬機(jī)版本,然后class文件要滿(mǎn)足虛擬機(jī)的規(guī)范。
-
準(zhǔn)備,需要準(zhǔn)備什么呢?
就是要進(jìn)行類(lèi)成員的初始化為初始值,其中為final修飾的類(lèi)變量除外,final變量就直接初始化為變量值,而類(lèi)成員不一樣。 -
解析,什么是解析呢?
就是把符號(hào)引用解析為直接引用,就是我們變量xxx,這種代表變?yōu)橹苯右?#xff0c;什么是直接引用呢?就是內(nèi)存地址,如我們常見(jiàn)的xxx0203r0e,這種。 -
初始化,把關(guān)于static修飾的變量或者是static靜態(tài)代碼塊按照順序組成構(gòu)造器進(jìn)行初始化變量。
-
使用,
-
卸載
當(dāng)類(lèi)加載到內(nèi)存后,jvm會(huì)將class常量池中的內(nèi)容存放到運(yùn)行時(shí)常量池中,所以運(yùn)行時(shí)常量池每個(gè)類(lèi)都有一個(gè)的。
class常量池是存放字面量和符號(hào)的引用,是對(duì)象的符號(hào)引用值,經(jīng)過(guò)解析就是把符號(hào)引用解析為直接引用,在編譯階段存放的是常量的符號(hào)引用,進(jìn)行解析后就是直接引用了。然后在全局常量池中保證每個(gè)jvm只有一份,存放的是字符串常量的直接引用值。
如果改為`string1 == string3.intern()`結(jié)果為true,因?yàn)榉祷氐氖浅A砍乩锩孀置嬷档牡刂贰?/p>
String類(lèi)的intern()方法,會(huì)在常量池中查找是否有一份equal()相等的字符串。
String string1 = "dashu"; String string3 = ?new String("dashu"); System.out.println(string1==string3.intern());如果常量池中沒(méi)有這個(gè)“dashu”字面量,那么就先把這個(gè)字面量“dashu”值,先放入到常量表之后,再返回常量表的地址。
常量池優(yōu)點(diǎn)
常量池可以避免因頻繁的創(chuàng)建和銷(xiāo)毀對(duì)象,從而導(dǎo)致系統(tǒng)性能的降低,也實(shí)現(xiàn)了對(duì)象的共享,即可以節(jié)省內(nèi)存空間,也可以節(jié)省運(yùn)行的時(shí)間。
總結(jié)
以上是生活随笔為你收集整理的Java当中的常量池的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 如何在 JS 代码中消灭 for 循环
- 下一篇: Java 中的四种引用类型