通過實體類的名稱字符串獲取字符串所表達的類對象:
String sb = "com.liuf.User";
T entity = (T) Class.forName(sb).newInstance();
Class對象的生成方式如下:
1.類名.class 說明: JVM將使用類裝載器, 將類裝入內存(前提是:類還沒有裝入內存),不做類的初始化工作.返回Class的對象
2.Class.forName(“類名字符串”) (注:類名字符串是包名+類名) 說明:裝入類,並做類的靜態初始化,返回Class的對象
3.實例對象.getClass() 說明:對類進行靜態初始化、非靜態初始化;返回引用運行時真正所指的對象(因為:子對象的引用可能會賦給父對象的引用變量中)所屬的類的Class的對象
結論:
我們知道,靜態的方法屬性初始化,是在加載類的時候初始化。而非靜態方法屬性初始化,是new類實例對象的時候加載。
因此,這段程序說明,三種方式生成Class對象,其實只有一個Class對象。在生成Class對象的時候,首先判斷內存中是否已經加載。
所以,生成Class對象的過程其實是如此的:
當我們編寫一個新的java類時,JVM就會幫我們編譯成class對象,存放在同名的.class文件中。在運行時,當需要生成這個類的對象,JVM就會檢查此類是否已經裝載內存中。若是沒有裝載,則把.class文件裝入到內存中。若是裝載,則根據class文件生成實例對象。
public static void main(String[] args){ try { //測試.class Class testTypeClass=TestClassType.class; System.out.println("testTypeClass---"+testTypeClass); //測試Class.forName()(***代表包路徑) Class testTypeForName=Class.forName("***.TestClassType"); System.out.println("testTypeForName---"+testTypeForName); //測試Object.getClass() TestClassType testTypeGetClass= new TestClassType(); System.out.println("testTypeGetClass---"+testTypeGetClass.getClass()); } catch (ClassNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
public class TestClassType { //構造函數 public TestClassType(){ System.out.println("----構造函數---"); } //靜態的參數初始化 static{ System.out.println("---靜態的參數初始化---"); } //非靜態的參數初始化 { System.out.println("----非靜態的參數初始化---"); } }
說明:main函數里的代碼可以分開執行看一下,思考理解一下。
補充:
java生成對象有多種方式,這里只針對上文說明的三種生成class對象方式來反向創建實體類。
通過class對象轉換成對應實體類,這里使用newInstance()函數,不同的是生成class方式不同,調用newInstance()之后返回類型不同
Object obj1=testTypeClass.newInstance(); TestClassType o1 = (TestClassType) obj1; Object obj2=testTypeForName.newInstance(); TestClassType o2 = (TestClassType) obj2; TestClassType o3=testTypeGetClass.getClass().newInstance();