java 反射 构造方法_Java反射之构造方法反射
上一篇Java反射之Class類我們介紹了java反射的關鍵類Class,
反射就是由一個java類映射得到一個java類。
所以,我們自然能想到,一個類中應該有哪些屬性,這里做個比方,人有名字年齡等屬性,所有一個Person類中就會有name,age這兩個屬性,同理,一個類中也有構造方法,字段,方法等屬性。
接下來我們首先了解構造方法,通過反射構造方法動態創建一個對象。
[一] Constructor類
Constructor對象代表的是一個構造方法,那么這個對象應該有哪些方法呢,我們猜測一定會有獲得這個構造方法的名稱,獲得該構造方法屬于的類,通過這個構造方法產生一個對象等方法。
從jdk文檔里可以看到有getName()獲得構造方法名稱,getDeclaringClass()獲得所屬類的類,newInstance()創建對象。
在開發中,我們常用Constructor類的newInstance()方法來動態創建對象。
[二] 獲得Constructor對象
在演示Constructor類中的方法前,先介紹如何獲得Constructor,和Class一樣,Constructor也不可以直接new出。
上面是Class類的jdk文檔中的兩個方法,這兩個方法分別是獲得指定的構造方法和獲得所有的構造方法。
所以我們知道,想要得到Constructor對象,就要先獲得Class對象。下面我們來演示以下
[三] 演示
我們依舊拿String這個類做演示。
從上圖可以看出,String類中有很多的構造方法。
栗子演示
1.獲得所有的構造方法
public static void main(String[] args) throws Exception {
Class cls = Class.forName("java.lang.String");
//獲取所有構造方法
Constructor[] constructor = cls.getConstructors();
for(Constructor cons : constructor){
System.out.println(cons);
}
}
結果:
public java.lang.String(byte[],int)
public java.lang.String(byte[],int,int,java.nio.charset.Charset)
public java.lang.String(java.lang.StringBuffer)
public java.lang.String(byte[])
public java.lang.String(byte[],int,int)
public java.lang.String(byte[],java.nio.charset.Charset)
public java.lang.String(byte[],java.lang.String) throws java.io.UnsupportedEncodingException
public java.lang.String()
public java.lang.String(java.lang.StringBuilder)
public java.lang.String(byte[],int,int,int)
public java.lang.String(int[],int,int)
public java.lang.String(char[],int,int)
public java.lang.String(char[])
public java.lang.String(java.lang.String)
public java.lang.String(byte[],int,int,java.lang.String) throws java.io.UnsupportedEncodingException
從輸出的情況可以猜測,Constructor類一定有方法,可以獲得構造方法前的修飾符,參數,拋出的異常等等.
2.獲得單個構造方法
public static void main(String[] args) throws Exception {
Class cls = Class.forName("java.lang.String");
//獲取單個構造方法
Constructor constructor = cls.getConstructor(StringBuffer.class);
String constructorName = constructor.getName();
System.out.println(constructorName);
}
結果:
java.lang.String
就像java的重載方法一樣,判斷方法重載的依據是參數個數的不同和參數類型的不同,所以獲得特定構造方法也是根據參數的類型,參數的個數來得到對應的構造方法。
將原方法參數的class類型作為getConstructor方法的參數傳入,獲得對應參數的構造方法。同理獲得其他構造方法也是相同的。
3.通過構造方法創建對象
public static void main(String[] args) throws Exception {
Class cls = Class.forName("java.lang.String");
//獲取單個構造方法
Constructor constructor = cls.getConstructor(StringBuffer.class);
//由于在編譯時,無法檢測到是什么類型的Constructor所以需要進行類型強轉
String str = (String)constructor.newInstance(new StringBuffer("abc"));
//拿到反射出來的對象,調用charAt方法測試
System.out.println(str.charAt(2));
}
結果:
c
這里有些朋友可能會出現這樣的錯誤:
public static void main(String[] args) throws Exception {
Class cls = Class.forName("java.lang.String");
//獲取單個構造方法
Constructor constructor = cls.getConstructor(StringBuffer.class);
//由于在編譯時,無法檢測到是什么類型的Constructor所以需要進行類型強轉
String str = (String)constructor.newInstance("abc");
//拿到反射出來的對象,調用charAt方法測試
System.out.println(str.charAt(2));
}
結果:
Exception in thread "main" java.lang.IllegalArgumentException: argument type mismatch
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:57)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
at java.lang.reflect.Constructor.newInstance(Constructor.java:526)
at club.leyvan.muzile.ConstructDemo.main(ConstructDemo.java:12)
類型不批配錯誤,這就是說,上面代碼中的constructor對應的是參數是StringBuffer的這個構造方法,而傳進去的參數不是StringBuffer對象,所以會報錯。
小結: 獲得方法時要用到類型class,調用方法時要用到上面相同類型的實例對象。
[四] Class中的newInstance()方法
在Class類中也有newInstance方法,查看Class類源碼就可以知道,該方法調用的時默認的無參構造方法來創建對象,采用緩存機制,將無參的構造方法緩存起來,當調用newInstance方法時,從緩存中取出無參構造方法創建對象。
和Constructor中的區別在于,Class中的newInstance方法調用的是無參構造,不能調用有參構造。
總結
以上是生活随笔為你收集整理的java 反射 构造方法_Java反射之构造方法反射的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: DataGrid中添加背景
- 下一篇: Java中从Orcle里取出数据时,为什