“反射”(Reflection)能夠讓運行于JVM中的程序檢測和修改運行時的行動。
反射帶來的好處包括:
聲明以下:
public Method getDeclaredMethod(String name, Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
解釋:
返回1個Method對象,該對象反應此Class對象所表示的類或接口的指定已聲明方法。
1. name : 是1個String,它指定所需方法的簡稱。
2. parameterTypes:是1個Class對象的變長數組,它按聲明順序標識該方法的形參類型。
注意:
getDeclaredMethod獲得該類聲明的public方法或protected方法,但是不包括繼承的方法。
聲明以下:
public Method getMethod(String name, Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
解釋:
返回1個Method對象,該對象反應此Class對象所表示的類或接口的指定公共成員方法。
1. name : 是1個String,它指定所需方法的簡稱。
2. parameterTypes:是1個Class對象的變長數組,它按聲明順序標識該方法的形參類型。
name參數就不需要解釋了,就是調用類的方法名稱。
可能很多同學剛接觸這個方法的時候,會對parameterTypes參數產生疑問,例如這個參數為何是Class泛型變長數組,其實舉個例子就很好理解了。
假定我們要反射的方法有4個參數,函數原型以下:
public void printInfo(String str, int iNum, double dNum, long i);
那我們通過返回獲得這個Method對象的時候,傳的parameterTypes以下所示:
getMethod("printInfo", String.class, int.class, double.class, long.class);
所以,parameterTypes其實就是對方法形參的類型抽象。
聲明以下:
public Object invoke(Object obj, Object... args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException
解釋:
Method類的invoke(Object obj, Object… args)方法接收的參數必須為對象。其中:
1. obj : 從中調用底層方法的對象。
2. args :用于方法調用的參數。
我們知道,Android有些類是沒有在SDK中開放的,例如你需要獲得系統屬性,需要調用到SystemProperties類的get方法,但是這個類并沒有在SDK中公然,我們可以在Android源碼中查看1下這個類:
package android.os;
import java.util.ArrayList;
import android.util.Log;
/**
* Gives access to the system properties store. The system properties
* store contains a list of string key-value pairs.
*
* {@hide}
*/
public class SystemProperties
{
// 省略具體實現代碼
/**
* Get the value for the given key.
* @return an empty string if the key isn't found
* @throws IllegalArgumentException if the key exceeds 32 characters
*/
public static String get(String key) {
if (key.length() > PROP_NAME_MAX) {
throw new IllegalArgumentException("key.length > " + PROP_NAME_MAX);
}
return native_get(key);
}
}
可以看到,這個前面有1個@hide標簽,所以這個類是沒法直接在代碼中調用的。
但是,在Android利用中,很多時候我們需要獲得得手機類型屬性(ro.product.model)。所以,這個時候,我們就需要在利用層反射SystemProperties類,調用get方法。具體實現源碼以下:
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import android.util.Log;
public class SystemProperties {
public static String get(String key) {
String value = "";
Class<?> cls = null;
try {
cls = Class.forName("android.os.SystemProperties");
Method hideMethod = cls.getMethod("get", String.class);
Object object = cls.newInstance();
value = (String) hideMethod.invoke(object, key);
} catch (ClassNotFoundException e) {
Log.e("zhengyi.wzy", "get error() ", e);
} catch (NoSuchMethodException e) {
Log.e("zhengyi.wzy", "get error() ", e);
} catch (InstantiationException e) {
Log.e("zhengyi.wzy", "get error() ", e);
} catch (IllegalAccessException e) {
Log.e("zhengyi.wzy", "get error() ", e);
} catch (IllegalArgumentException e) {
Log.e("zhengyi.wzy", "get error() ", e);
} catch (InvocationTargetException e) {
Log.e("zhengyi.wzy", "get error() ", e);
}
return value;
}
}