Java程序在運行時,Java運行時系統一直對全部的對象進行所謂的運行時類型標識,即所謂的RTTI。這項信息紀錄了每一個對象所屬的類。虛擬機一般使用運行時類型信息選準正確方法去執行,用來保存這些類型信息的類是Class類。Class類封裝一個對象和接口運行時的狀態,當裝載類時,Class類型的對象自動建立。java
Class類的做用是運行時提供或得到某個對象的類型信息,和C++中的typeid()函數相似。這些信息也可用於反射。函數
咱們都知道全部的java類都是繼承了object這個類,在object這個類中有一個方法:getclass().這個方法是用來取得該類已經被實例化了的對象的該類的引用,這個引用指向的是Class類的對象。咱們本身沒法生成一個Class對象(構造函數爲private),而 這個Class類的對象是在當各種被調入時,由 Java 虛擬機自動建立 Class 對象,或經過類裝載器中的 defineClass 方法生成。咱們生成的對象都會有個字段記錄該對象所屬類在CLass類的對象的所在位置。以下圖所示:
code
第一種辦法,Class類的forName函數對象
public class shapes{} Class obj= Class.forName("shapes");
第二種辦法,使用對象的getClass()函數繼承
public class shapes{} shapes s1=new shapes(); Class obj=s1.getClass(); Class obj1=s1.getSuperclass();//這個函數做用是獲取shapes類的父類的類型
第三種辦法,使用類字面常量接口
Class obj=String.class; Class obj1=int.class;
注意,使用這種辦法生成Class類對象時,不會使JVM自動加載該類(如String類)。而其餘辦法會使得JVM初始化該類。get
獲取一個Class類的對象後,能夠用 newInstance() 函數來生成目標類的一個實例。然而,該函數並不能直接生成目標類的實例,只能生成object類的實例虛擬機
Class obj=Class.forName("shapes"); Object ShapesInstance=obj.newInstance();
Class<shapes> obj=shapes.class; shapes newShape=obj.newInstance();
由於有了類型限制,因此使用泛化Class語法的對象引用不能指向別的類。class
Class obj1=int.class; Class<Integer> obj2=int.class; obj1=double.class; //obj2=double.class; 這一行代碼是非法的,obj2不能改指向別的類了
然而,有個靈活的用法,使得你能夠用Class的對象指向基類的任何子類。泛型
Class<? extends Number> obj=int.class; obj=Number.class; obj=double.class;
所以,如下語法生成的Class對象能夠指向任何類。
Class<?> obj=int.class; obj=double.class; obj=shapes.class;
最後一個奇怪的用法是,當你使用這種泛型語法來構建你手頭有的一個Class類的對象的基類對象時,必須採用如下的特殊語法
public class shapes{} class round extends shapes{} Class<round> rclass=round.class; Class<? super round> sclass= rclass.getSuperClass(); //Class<shapes> sclass=rclass.getSuperClass();
咱們明知道,round的基類就是shapes,可是卻不能直接聲明 Class < shapes >,必須使用特殊語法
Class < ? super round >
這個記住就能夠啦。