Java反射機(jī)制
JAVA反射機(jī)制是在運行狀態(tài)中,對于任意一個類,都能夠知道這個的所有屬性和方法;對于任意一個對象擅耽,都能夠調(diào)用它的任意一個方法;這種動態(tài)獲取的信息以及動態(tài)調(diào)用對象的方法的功能成為java語言的反射機(jī)制物遇。
反射機(jī)制主要提供了如下幾種功能:
1乖仇、在運行時判斷任意一個對象所屬的類
2、在運行時構(gòu)造任意一個類的對象
3询兴、在運行時判斷任意一個類所具有的成員變量和方法
4乃沙、在運行時調(diào)用任意一個對象的方法
5、生成動態(tài)代理
代碼示例:
1.得到某個對象的屬性
public Object getProperty(Object owner, String fieldName) throws Exception {
Class ownerClass = owner.getClass();
Field field = ownerClass.getField(fieldName);
Object property = field.get(owner);
return property;
}
Class ownerClass = owner.getClass():得到該對象的Class诗舰。
Field field = ownerClass.getField(fieldName):通過Class得到類聲明的屬性警儒。
Object property = field.get(owner):通過對象得到該屬性的實例,如果這個屬性是非公有的眶根,這里會報IllegalAccessException蜀铲。
2.得到某個類的靜態(tài)屬性:
public Object getStaticProperty(String className, String fieldName)
throws Exception {
Class ownerClass = Class.forName(className);
Field field = ownerClass.getField(fieldName);
Object property = field.get(ownerClass);
return property;
}
Class ownerClass = Class.forName(className) :首先得到這個類的Class边琉。
Field field = ownerClass.getField(fieldName):和上面一樣,通過Class得到類聲明的屬性蝙茶。
Object property = field.get(ownerClass) :這里和上面有些不同艺骂,因為該屬性是靜態(tài)的,所以直接從類的Class里取隆夯。
3.執(zhí)行某對象的方法:
public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
Class ownerClass = owner.getClass();
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(owner, args);
}
Class owner_class = owner.getClass() :首先還是必須得到這個對象的Class钳恕。
5~9行:配置參數(shù)的Class數(shù)組,作為尋找Method的條件蹄衷。
Method method = ownerClass.getMethod(methodName, argsClass):通過methodName和參數(shù)的argsClass(方法中的參數(shù)類型集合)數(shù)組得到要執(zhí)行的Method忧额。
method.invoke(owner, args):執(zhí)行該Method.invoke方法的參數(shù)是執(zhí)行這個方法的對象owner,和參數(shù)數(shù)組args愧口,可以這么理解:owner對象中帶有參數(shù)args的method方法睦番。返回值是Object,也既是該方法的返回值耍属。
4.執(zhí)行某個類的靜態(tài)方法:
public Object invokeStaticMethod(String className, String methodName,
Object[] args) throws Exception {
Class ownerClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(null, args);
}
基本的原理和實例3相同托嚣,不同點是最后一行,invoke的一個參數(shù)是null厚骗,因為這是靜態(tài)方法示启,不需要借助實例運行。
5.新建實例:
public Object newInstance(String className, Object[] args) throws Exception {
Class newoneClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Constructor cons = newoneClass.getConstructor(argsClass);
return cons.newInstance(args);
}
這里說的方法是執(zhí)行帶參數(shù)的構(gòu)造函數(shù)來新建實例的方法领舰。如果不需要參數(shù)夫嗓,可以直接使用newoneClass.newInstance()來實現(xiàn)。
Class newoneClass = Class.forName(className):第一步冲秽,得到要構(gòu)造的實例的Class舍咖。
第5~第9行:得到參數(shù)的Class數(shù)組。
Constructor cons = newoneClass.getConstructor(argsClass):得到構(gòu)造子锉桑。
cons.newInstance(args):新建實例排霉。
6.判斷是否為某個類的實例:
public boolean isInstance(Object obj, Class cls) {
return cls.isInstance(obj);
}
7.得到數(shù)組中的某個元素:
public Object getByArray(Object array, int index) {
return Array.get(array,index);
}