編輯:關於Android編程
反射定義
“反射”(Reflection)能夠讓運行於JVM中的程序檢測和修改運行時的行為。
為何需要反射
反射帶來的好處包括:
反射方法Method
getDeclaredMethod方法
聲明如下:
public Method getDeclaredMethod(String name, Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
解釋:
返回一個Method對象,該對象反映此Class對象所表示的類或接口的指定已聲明方法。
1. name : 是一個String,它指定所需方法的簡稱。
2. parameterTypes:是一個Class對象的變長數組,它按聲明順序標識該方法的形參類型。
注意:
getDeclaredMethod獲取該類聲明的public方法或者protected方法,但是不包括繼承的方法。
getMethod方法
聲明如下:
public Method getMethod(String name, Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
解釋:
返回一個Method對象,該對象反映此Class對象所表示的類或接口的指定公共成員方法。
1. name : 是一個String,它指定所需方法的簡稱。
2. parameterTypes:是一個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其實就是對方法形參的類型抽象。
invoke方法
聲明如下:
public Object invoke(Object obj, Object... args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException
解釋:
Method類的invoke(Object obj, Object… args)方法接收的參數必須為對象。其中:
1. obj : 從中調用底層方法的對象。
2. args :用於方法調用的參數。
Android 反射應用
我們知道,Android有些類是沒有在SDK中開放的,例如你需要獲取系統屬性,需要調用到SystemProperties類的get方法,但是這個類並沒有在SDK中公開,我們可以在Android源碼中查看一下這個類:
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); } }
可以看到,這個前面有一個@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; } }
最近到4412最基本的都調了 然後覺得沒事做了 所以增加一個HDMI的設置 不能閒著 以下使用的是廣州斯道的icool210開發板 源碼修改如下 最開始我是先在設置裡加上
1. 如何創建Looper?Looper的構造方法為private,所以不能直接使用其構造方法創建。private Looper(boolean quitAllowed)
今天和大家分享下組合控件的使用。很多時候android自定義控件並不能滿足需求,如何做呢?很多方法,可以自己繪制一個,可以通過繼承基礎控件來重寫某些環節,當
Android中的Service和其調用者既可以在同一個App中,也可以在不同的App。如果Service在App1中,而調用Service的客戶端在App2中,那麼我們