歡迎各位加入個人Android開發羣[257053751]java
Android開發中,有一個讓人又愛又恨的方法叫findViewById(int);我想若是你是一民Android開發者,必然知道這個方法,git
爲何說findViewById(int);讓人又愛又恨呢?想必你們也是頗有感觸。
寫一個佈局,用Java代碼寫和用xml文件寫,完成速度徹底是沒法比擬的。xml佈局太方便了。
一樣的,想獲取一個控件的對象,若是你是使用的xml佈局文件寫的佈局,那麼你必須調用findViewById()這個方法。
github
TextView t = (TextView) findViewById(R.id.x);
這是咱們最多見的 獲取xml佈局中一個textview對象的過程。
那麼問題就來了,這特麼奇葩的方法名也太長了吧!!!好吧,其實人家名字起的也沒有錯,要描述清楚這函數的含義,也必須這麼多個字母。express
但是你丫的返回一個View讓我用的時候還得強轉,這也太麻煩了吧。我一行代碼總共也就80列(Eclipse默認),縮進八格(方法寫在類裏面,語句寫在方法裏面),
就算像上面的例子textView對象只有一個字母,id也只有一個字母,這一個初始化也要佔我54列了。要是變量名再長點,縮進層次再深點,這一個初始化就兩行了。
一個界面至少也有四個控件吧,這麼複雜的初始化,太坑爹了。
有問題總會有對應的解決辦法,下面我就向你們介紹一下KJFrameForAndroid框架使用註解解決這種麻煩。框架
KJFrameForAndroid框架項目地址:https://github.com/kymjs/KJFrameForAndroid。函數
從jdk1.5開始,Java提供了註解的功能,容許開發者定義和使用本身的註解類型,該功能由一個定義註解類型的語法和描述一個註解聲明的語法,讀取註解的API,一個使用註解修飾的class文件和一個註解處理工具組成。
首先,你須要接受一個關鍵字@interface ,噢,它可不是接口定義關鍵字,更不是OC裏面的@interface關鍵字,是Java中表示聲明一個註解類的關鍵字。
使用@interface 表示咱們已經繼承了java.lang.annotation.Annotation類,這是一個註解的基類接口,就好像Object類,如今你只須要知道它的存在就好了。
還有一條規定:在定義註解時,不能繼承其餘的註解或接口。
那麼,這就是最簡單的一個註解類
工具
public @interface MyAnnotation { }
然而一般在使用時咱們都會給這個註解類加上兩個註解:佈局
@Target(ElementType.FIELD)、@Retention(RetentionPolicy.RUNTIME)
ElementType、RetentionPolicy是兩個枚舉類,ElementType.FIELD表示咱們須要註解的是一個字段,如下是摘自JDK1.6文檔中的介紹:this
如下爲KJFrameForAndroid框架中綁定控件註解部分的定義與使用:spa
@Target(ElementType.FIELD) @Retention(RetentionPolicy.RUNTIME) public @interface BindView { public int id(); public boolean click() default false; }
@BindView(id = R.id.x, click = true) private TextView t;
咱們能夠看到,除了明顯減小了代碼量,還使得代碼結構更加清晰。
其中,定義部分的id() 表示註解接受一個int類型的數據做爲id所對應的值(就如使用中的id = R.id.xxx);
同理,定義部分的click表示接受一個Boolean類型的數據做爲click對應的值,還能夠設置一個默認值使用default修飾;
咱們已經知道了註解怎麼定義和使用,接下來就應該知道怎麼處理了。
上面已經說了,bindview註解能夠接受一個int類型的值和一個Boolean類型的值,那麼這兩個值接受了之後如何獲取呢?
其實獲取的方式很簡單就是經過一個BindView類型的對象,調用這個對象來自聲明中定義的兩個方法——>id()或click()方法。
如今就有一個問題了,註解類型是不能直接new對象的,那麼這個BindView對象從哪裏來呢?
這時就須要用到Java的反射機制。咱們知道,每個繼承自Object類的類都會繼承一個getClass()方法,下面看一下這個方法的原型:
/** * Returns the unique instance of {@link Class} that represents this * object's class. Note that {@code getClass()} is a special case in that it * actually returns {@code Class<? extends Foo>} where {@code Foo} is the * erasure of the type of the expression {@code getClass()} was called upon. * <p> * As an example, the following code actually compiles, although one might * think it shouldn't: * <p> * <pre>{@code * List<Integer> l = new ArrayList<Integer>(); * Class<? extends List> c = l.getClass();}</pre> * * @return this object's {@code Class} instance. */ public final native Class<?> getClass();
是一個native方法,根據註釋咱們知道,這個方法返回的是該類的Class對象,同時也是該類的二進制對象。
Class中有一個方法叫getDeclaredFields(),是用來返回這個類的所有字段,返回類型是Field[]
經過Field對象的getAnnotation(Class<?>)方法,咱們能夠獲取到任何一個Class的對象,經過getAnnotation(Class<?>),咱們就能夠獲取到BindView的對象了。
例如:
Field[] fields = currentClass.getClass().getDeclaredFields(); for(int i = 0; i < fields.length; i++){ BindView bindView = field.getAnnotation(BindView.class); int viewId = bindView.id(); //這是咱們傳的id boolean clickLis = bindView.click(); //這是咱們傳的click }
至此,咱們已經瞭解了註解,而且知道怎麼使用,怎麼處理註解了,如今只剩下最後一個問題:在項目中使用。
很簡單,傳一個Activity對象,調用findViewById()不就好了。
因而,咱們能夠這樣
activity.findViewById( bindView.id() );
最後在咱們的Activity中調用這個函數就OK了。
如下是Android應用框架KJFrameForAndroid中使用註解綁定控件的核心代碼:
/** * @param currentClass * 當前類,通常爲Activity或Fragment * @param sourceView * 待綁定控件的直接或間接父控件 */ public static void initBindView(Object currentClass, View sourceView) { // 經過反射獲取到所有屬性,反射的字段多是一個類(靜態)字段或實例字段 Field[] fields = currentClass.getClass().getDeclaredFields(); if (fields != null && fields.length > 0) { for (Field field : fields) { // 返回BindView類型的註解內容 BindView bindView = field.getAnnotation(BindView.class); if (bindView != null) { int viewId = bindView.id(); boolean clickLis = bindView.click(); try { field.setAccessible(true); if (clickLis) { sourceView.findViewById(viewId).setOnClickListener( (OnClickListener) currentClass); } // 將currentClass的field賦值爲sourceView.findViewById(viewId) field.set(currentClass, sourceView.findViewById(viewId)); } catch (Exception e) { e.printStackTrace(); } } } } }
其實安卓中的註解式綁定控件(也是所謂的IOC控制反轉在安卓中的一種應用)其實本質的使用就是Java基礎中反射的使用。值得一提的是,反射執行的效率是很低的
若是不是必要,應當儘可能減小反射的使用,由於它會大大拖累你應用的執行效率。順帶一提:我一直很排斥註解,由於類反射的效率過低了。如今有不少安卓應用開發框架,好比KJFrameForAndroid, xUtils, afinal, thinkAndroid,這些框架都是使用反射來起到註解綁定控件。更有的框架甚至是一切東西都使用註解去完成,我只能說註解便捷,但請慎用。