本篇文章將分為兩部分:
1、通過反射獲取類的構(gòu)造方法
獲取某一個(gè)類中的所有的構(gòu)造器,通常分為兩個(gè)步驟:
①明確操作的是哪一份字節(jié)碼對(duì)象
②獲取構(gòu)造器
Class類獲取構(gòu)造器方法:
Constructor類:表示類中構(gòu)造器的類型,Constructor的實(shí)例就是某一個(gè)類中的某一個(gè)構(gòu)造器
public Constructor<?>[] getConstructors():
該方法只能獲取當(dāng)前Class所表示類的public修飾的構(gòu)造器
public Constructor<?>[] getDeclaredConstructors():
獲取當(dāng)前Class所表示類的所有的構(gòu)造器,和訪問權(quán)限無關(guān)
public Constructor<T> getConstructor(Class<?>... parameterTypes)
:獲取當(dāng)前Class所表示類中指定的一個(gè)public的構(gòu)造器
參數(shù):parameterTypes表示:構(gòu)造器參數(shù)的Class類型
如:
public User(String name) Constructor c = clz.getCOnstructor(String.class);
public Constructor<T> getDeclaredConstructor(Class<?>...parameterTypes)
:獲取當(dāng)前Class所表示類中指定的一個(gè)的構(gòu)造器,和訪問權(quán)限無關(guān)
class User {
public User() {}
public User(String name) {}
private User(String name, int age) {}
}
public class ConstructorDemo {
public static void main(String[] args) throws Exception {
//1、先找到被調(diào)用構(gòu)造器所在類的字節(jié)碼
Class<User> clz = User.class;
//2、獲取所有構(gòu)造器
Constructor<?>[] cs = clz.getConstructors();
for (Constructor<?> constructor : cs) {
System.out.println(constructor);
}
System.out.println("---------------------------");
//2、獲取指定公有帶參數(shù)構(gòu)造器(獲取指定公有構(gòu)造器)
Constructor<?> c = clz.getConstructor(String.class);
System.out.println(c);
System.out.println("---------------------------");
//獲取指定公有無參數(shù)構(gòu)造器(獲取指定公有構(gòu)造器)
c = clz.getConstructor();
System.out.println(c);
System.out.println("---------------------------");
//2、獲取指定私有構(gòu)造器
c = clz.getDeclaredConstructor(String.class,int.class);
System.out.println(c);
}
}
當(dāng)我們?cè)诖a中添加如下代碼:
c = clz.getDeclaredConstructor(int.class);
System.out.println(c);
表示:在User中沒有找到帶有int類型的構(gòu)造器
2、創(chuàng)建對(duì)象
調(diào)用構(gòu)造器,創(chuàng)建對(duì)象:
Constructor<T>
類:表示類中構(gòu)造器的類型,Constructor
的實(shí)例就是某一個(gè)類中的某一個(gè)構(gòu)造器
常用方法:
public T newInstance(Object...initargs)
:如調(diào)用帶參數(shù)的構(gòu)造器,只能使用該方式.
參數(shù):initargs
:表示調(diào)用構(gòu)造器的可變實(shí)際參數(shù)
返回:返回創(chuàng)建的實(shí)例,T
表示Class
所表示類的類型
如果:一個(gè)類中的構(gòu)造器可以直接訪問,同時(shí)沒有參數(shù).,那么可以直接使用Class類中的newInstance
方法創(chuàng)建對(duì)象.
public Object newInstance()
:相當(dāng)于new 類名();
class User {
public User() {
System.out.println("User()");
}
public User(String name) {
System.out.println("User(String name) " + "name = "+name);
}
private User(String name, int age) {
System.out.println("User(String name, int age) " + "name = "+name+" age = "+age);
}
}
public class ConstructorDemo {
public static void main(String[] args) throws Exception {
//1、先找到被調(diào)用構(gòu)造器所在類的字節(jié)碼
Class<User> clz = User.class;
//2、獲取指定公有帶參數(shù)構(gòu)造器(獲取指定公有構(gòu)造器)
Constructor<?> c = clz.getConstructor(String.class); //
//3、實(shí)例化對(duì)象
c.newInstance("huangweiyong");
System.out.println("---------------------------");
//獲取指定公有無參數(shù)構(gòu)造器(獲取指定公有構(gòu)造器)
c = clz.getConstructor();
//3、實(shí)例化對(duì)象
c.newInstance();
System.out.println("---------------------------");
//直接調(diào)用Class類的newInstance方法構(gòu)造對(duì)象
//3、實(shí)例化對(duì)象
clz.newInstance();
System.out.println("---------------------------");
//、獲取指定私有構(gòu)造器
c = clz.getDeclaredConstructor(String.class,int.class);
//設(shè)置私用方法的可訪問(切記,這里必須設(shè)置,否則會(huì)拋出下圖的異常)
c.setAccessible(true);
//3、實(shí)例化對(duì)象
c.newInstance("huangweiyong",18);
}
}
c.setAccessible(true);
如果上述代碼在調(diào)用私有構(gòu)造函數(shù)時(shí)沒有設(shè)置成true
或者不寫,會(huì)拋出異常:
上述異常的原因是:不能夠訪問User類中一個(gè)修飾符為private的成員.
為了安全性考慮,private成員外界不準(zhǔn)訪問.
但是問題是,我偏要訪問:此時(shí)就要告訴成員在運(yùn)行期間忽略掉該安全檢查.
解決方案:調(diào)用AccessibleObject
類中的
publicvoid setAccessible(boolean flag)
方法:設(shè)置是否可訪問
setAccessible(true)
:可訪問的
又因?yàn)?code>Constructor是AccessibleObject
子類,所以Constructor
中具有該方法.
(這一點(diǎn)千萬不要忘記,切記切記!!!!!!)
作者:老勇
鏈接:http://www.lxweimin.com/p/60e622123c7c
來源:簡(jiǎn)書
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請(qǐng)注明出處。