LocalVariableTable,LocalVariableTypeTable
1.「LocalVariableTable」
?用于描述局部變量表中的變量與JAVA代碼中定義變量之間的關系,同樣也可以選擇不生成該屬性
?
?用處:當編寫代碼引用到這個方法時,參數可以直接顯示變量名和類型,如果沒有該屬性,就用arg0,arg1代替;調試信息的時候可以根據參數變量名明確語義。
?
2.「LocalVariableTypeTable」
?引入泛型后加入的屬性,結構和LocalVariableTable相似,將原先用于描述字段描述符的descriptor_index替換成了描述字段的特征簽名(Signature)。
?
?對于非泛型變量,特征簽名和描述符是一致的,但是對于泛型來說,由于泛型參數類型的擦除,描述符不能夠描述泛型類型,因此定義了該屬性來完成泛型的描述。
?
結構:
ConstantValue
「變量初始化,賦值時機:」 位于該屬性結構中的常量將會在類加載的準備階段就會初始化并且賦值;
其他的靜態變量在這個階段只是會被初始化然后賦默認值,如果靜態變量設置了final關鍵字,那么就是第一種情況會對變量進行賦值;
對于實例變量(非靜態變量)的賦值是在實例構造器《init》中。
「該結構中存放的字段是:」
「《Java虛擬機規范》中規定該屬性結構中存放的必須是靜態的字段,而對于javac編譯器來說還需要滿足final關鍵字的修飾,因此經過javac編譯器編譯后的該屬性中存放的字段必須是static并且是final的。」
?通知虛擬機自動為靜態變量賦值(上面那句話)。該屬性中只能存放基本類型和String,因為該屬性的屬性值
?
結構:
Deprecated及Synthetic屬性
?這兩項屬性有點特殊,不攜帶任何屬性值,出現這兩個屬性的目的只是為了標識,這兩個屬性只有存在或不存在。
?
- 「Deprecated屬性」 該屬性用于表示某個類,字段或方法已經不再推薦使用,通過“@deprecated”注解設置這個屬性
2.「Synthetic屬性」
該屬性用于表示字段或者方法是編譯器自己添加的,不是代碼中的。也可以通過設置訪問標志ACC_SYNTHETIC標志位生成該項屬性。
結構:兩者都一樣,不攜帶任何屬性值。只是用于標識
StackMapTable
「該屬性位于Code屬性的屬性表中」
?用處:在類加載階段的驗證階段使用該屬性,代替以前耗性能的基于數據流分析的類型推導驗證器(有了該屬性之后就不用類型推導了,可以直接判斷類型是不是符合要求,之后單獨寫類加載階段進行分析)
?
?之前驗證階段是基于數據流來進行分析推導出操作數棧和本地變量表操作的類型是否一致等(比如istore需要將操作數棧的數據保存到本地變量表中,但是取出的數據類型不是int就會發生問題),現在基于該項屬性可以不用推導
?
結構:「一個Code屬性最多只能有一個StackMapTable屬性」
MethodParameters
「用于記錄方法的各個形參名稱和信息」
方法參數屬性,位于class中的屬性表中。之前說過這部分是存儲在局部變量表中的,因為方法中有方法體code屬性,而code中需要有局部變量表屬性代表這個方法中的變量存儲。
但是為什么還要單獨抽出一個屬性放在class中呢?
大家想想沒有code就沒有局部變量表,沒有局部變量表是不是就不能存儲方法參數了;你看接口中他有方法吧但是呢他其實沒有方法提code所以它的方法參數往哪放呢?往他借口的屬性表集合中放,也就是與code同級。這樣的話我接口里可以直接獲得方法參數通過這個屬性;而對于正常的方法也就是有方法體的代碼可以從code中的局部變量表中拿。
數據結構:1.首先說明他是什么,我是一個方法參數類型 2.我說明我存儲的時候數據有多長(多少字節),為了切割按照這個就可以正確讀取對應的數據;但是如果這個屬性中還用到了其他的數據結構(屬性),那么就是這個屬性的個數了
3.對于沒有再次用到其他屬性來描述的屬性直接使用定長數據即可;但是對于有用到其他屬性來描述這個屬性的話,則后面是對應的屬性一個一個排開,然后每個屬性如果是定長的話則不需要通過長度來說明所占字節,然后這個屬性中存儲的第一個永遠是他是什么也就是名字,然后再是對應的值
不斷使用這種結構來描述一個完整的class結構
結構:
-
JAVA
+關注
關注
19文章
2974瀏覽量
105018 -
Class
+關注
關注
0文章
53瀏覽量
19769 -
JVM
+關注
關注
0文章
158瀏覽量
12252
發布評論請先 登錄
相關推薦
評論