基本數據類型的載入和存儲 極客時間深入理解Java虛擬機讀後感,有錯誤還請指正 虛擬機中的Boolean類型 在Java語言規範中,boolean類型的值只有兩種可能,那就是"true"和"false". 但是這兩個符號是不能被Java虛擬機直接使用的. 在Java虛擬機規範中,boolean類型則 ...
基本數據類型的載入和存儲
極客時間深入理解Java虛擬機讀後感,有錯誤還請指正
虛擬機中的Boolean類型
在Java語言規範中,boolean類型的值只有兩種可能,那就是"true"和"false". 但是這兩個符號是不能被Java虛擬機直接使用的.
在Java虛擬機規範中,boolean類型則被映射成int類型. 也就是說,"true"被映射為整數1,"false"被映射為整數0.這個編碼規則約束了Java位元組碼的具體實現.
Java中的基本類型
Java中的基本類型有8個,上面的boolean類型, 整數類型的byte、short、char、int和 long,以及浮點類型 float 和 double.
java的基本類型都有對應的值域和預設值,如圖,從上到下的值域一次增大,後面的值域包含前面的值域,意味著上面的數據類型轉換為下麵的數據類型不需要進行強制轉換。還有註意預設值看起來不一樣,實際在記憶體中都是0。
8個基本數據類型中,只有char和boolean是無符號數。並且boolean類型的取值範圍為0或者1,char類型的取值範圍為[0,65535]。通常我們認為char類型的值為非負數。
Java 的浮點類型採用 IEEE 754 浮點數格式。以 float 為例,浮點類型通常有兩個 0,+0.0F 以及 -0.0F。前者在 Java 里是 0,後者是符號位為 1、其他位均為 0 的浮點數,在記憶體中等同於十六進位整數 0x8000000(即 -0.0F 可通過Float.intBitsToFloat(0x8000000) 求得)。儘管它們的記憶體
數值不同,但是在 Java 中 +0.0F == -0.0F 會返回真。
有了+0.0F和-0.0F後,那麼浮點數中的正無窮和負無窮就可以定義了。正無窮就是任意正浮點數(不包括 +0.0F)除以 +0.0F 得到的值,而負無窮是任意正浮點數除以 -0.0F得到的值。在 Java 中,正無窮和負無窮是有確切的值,在記憶體中分別等同於十六進位整數0x7F800000 和 0xFF800000。
那超出範圍的數字呢? 對應的是NaN(Not-a-Number),0x7FC00000為標準的NaN,其他的稱之為不標準的NaN。並且NaN有一個特性,任何數 != NaN 永遠返回true。
Java基本類型的大小
存儲時的大小
Java 虛擬機每調用一個 Java 方法,便會創建一個棧幀。暫時理解為解釋器使用的解釋棧幀。包括兩個組成部分,局部變數區和位元組碼操作數棧。
在Java虛擬機規範中,局部變數區等價於一個數組,並且可以用正整數索引。除了long、double值需要用兩個數組單元來存儲之外,其他的基本類型以及引用類型的值均占用一個數組單元。
也就是說,boolean、byte、char、short這四種類型,在棧上占用的空間和int是一樣的,和引用類型也是一樣的。也就是32位的HotSpot上,棧上占用4個位元組;64位的HotSpot上,占用8個位元組。
當然,這種情況僅存在於局部變數,而並不會出現在存儲於堆上的欄位和數組元素上。對於byte、char以及short這三種類型的欄位或者數組單元,他們在堆上占用的空間分別為一位元組、兩位元組以及兩位元組,與值域是相吻合的。
因此,當我們將一個 int 類型的值,存儲到這些類型的欄位或數組時,相當於做了一次隱式的掩碼操作。舉例來說,當我們把0xFFFFFFFF(-1)存儲到一個聲明為 char 類型的欄位里時,由
於該欄位僅占兩位元組,所以高兩位的位元組便會被截取掉,最終存入“\uFFFF”。
boolean 欄位和 boolean 數組則比較特殊。在 HotSpot 中,boolean 欄位占用一位元組,而boolean 數組則直接用 byte 數組來實現。為了保證堆中的 boolean 值是合法的,HotSpot 在存儲時顯式地進行掩碼操作,也就是說,只取最後一位的值存入boolean 欄位或數組中。
載入時的大小
Java 虛擬機的算數運算幾乎全部依賴於操作數棧。也就是說,我們需要將堆中的 boolean、byte、char 以及 short 載入到操作數棧上,而後將棧上的值當成 int 類型來運算。
對於 boolean、char 這兩個無符號類型來說,載入伴隨著零擴展。舉個例子,char 的大小為兩個位元組。在載入時 char 的值會被覆制到 int 類型的低二位元組,而高二位元組則會用 0 來填充。
對於 byte、short 這兩個類型來說,載入伴隨著符號擴展。舉個例子,short 的大小為兩個位元組。在載入時 short 的值同樣會被覆制到 int 類型的低二位元組。如果該 short 值為非負數,即最
高位為 0,那麼該 int 類型的值的高二位元組會用 0 來填充,否則用 1 來填充。
總結
其中,boolean 類型在 Java 虛擬機中被映射為整數類型:“true”被映射為 1,而“false”被映射為 0。Java 代碼中的邏輯運算以及條件跳轉,都是用整數相關的位元組碼來實現的。
除 boolean 類型之外,Java 還有另外 7 個基本類型。它們擁有不同的值域,但預設值在記憶體中均為 0。這些基本類型之中,浮點類型比較特殊。基於它的運算或比較,需要考慮+0.0F、-0.0F 以及 NaN 的情況。
除 long 和 double 外,其他基本類型與引用類型在解釋執行的方法棧幀中占用的大小是一致的,但它們在堆中占用的大小確不同。在將 boolean、byte、char 以及 short 的值存入欄位或
者數組單元時,Java 虛擬機會進行掩碼操作。在讀取時,Java 虛擬機則會將其擴展為 int 類型。
我的理解
從這個中,可以學到,8種基本數據的大小,還有虛擬機對於基本數據類型的規範是什麼,堆上和解釋執行的方法棧幀上的byte、short、char、boolean類型的存儲的大小是不一樣的,載入(讀取)的時候,會進行掩碼的操作。