大龙的博客

常用链接

统计

最新评论

【Android框架进阶〖00〗】ThinkAndroid注解机制 --- 转

由于项目需要,开始研究ThinkAndroid。

个人认为该框架的注解机制十分新颖,所以先研究这个,顺便学习下 Java 的annotation。

粗略的看了看,该机制在BaseActivity中初始化。而BaseActivity是所有Activity的基类。

对BaseActivity进行了代码剖离,发现在BaseActivity中在onCreate函数里启动注解机制。

  • 首先注入布局资源(绑定layout布局)
  • 其次注入成员资源(绑定组件资源)
  • 然后注入成员变量(初始化普通变量)

暂时先搞清楚第二个,怎么绑定组件资源的:

我写了个小demo,MyAnno

InjectView.java

  1. package com.myanno;  
  2.   
  3. import java.lang.annotation.ElementType;  
  4. import java.lang.annotation.Retention;  
  5. import java.lang.annotation.RetentionPolicy;  
  6. import java.lang.annotation.Target;  
  7.   
  8.   
  9. /** 
  10.  * 自定义注解,获取资源ID 
  11.  * @使用方法 
  12.  *              @InjectView(id = R.id.settingIv) 
  13.  *              private ImageView imgSetting;  
  14.  *  
  15.  * @author      michael.mao@sosino.com 
  16.  * @date        2013-10-29 
  17.  * @description @Retention: 定义注解的保留策略 
  18.  *              @Retention(RetentionPolicy.SOURCE)  //注解仅存在于源码中,在class字节码文件中不包含 
  19.  *              @Retention(RetentionPolicy.CLASS)   //默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得, 
  20.  *              @Retention(RetentionPolicy.RUNTIME) //注解会在class字节码文件中存在,在运行时可以通过反射获取到 
  21.  *              @Inherited                          //说明子类可以继承父类中的该注解 
  22.  * 
  23.  *              @Target(ElementType.TYPE)           //接口、类、枚举、注解 
  24.  *              @Target(ElementType.FIELD)          //字段、枚举的常量 
  25.  *              @Target(ElementType.METHOD)         //方法 
  26.  *              @Target(ElementType.PARAMETER)      //方法参数 
  27.  *              @Target(ElementType.CONSTRUCTOR)    //构造函数 
  28.  *              @Target(ElementType.LOCAL_VARIABLE) //局部变量 
  29.  *              @Target(ElementType.ANNOTATION_TYPE)//注解 
  30.  *              @Target(ElementType.PACKAGE)        //包    
  31.  */  
  32.   
  33. @Target(ElementType.FIELD)  
  34. @Retention(RetentionPolicy.RUNTIME)  
  35. public @interface InjectView  
  36. {  
  37.     /** View的ID */  
  38.     public int id() default -1;  
  39. }  

MainActivity.java
  1. package com.myanno;  
  2.   
  3. import java.lang.reflect.Field;  
  4.   
  5. import android.app.Activity;  
  6. import android.os.Bundle;  
  7. import android.util.Log;  
  8. import android.view.View;  
  9. import android.widget.ImageView;  
  10. import android.widget.TextView;  
  11.   
  12. public class MainActivity extends Activity {  
  13.   
  14.     /** 注解绑定UI元素 */  
  15.     @InjectView(id=R.id.myimgview)  
  16.     ImageView myimageview;  
  17.       
  18.     @InjectView(id=R.id.mytext)  
  19.     TextView mytext;  
  20.   
  21.     @Override  
  22.     protected void onCreate(Bundle savedInstanceState) {  
  23.         super.onCreate(savedInstanceState);  
  24.         setContentView(R.layout.activity_main);  
  25.           
  26.         //初始化注解绑定的成员变量  
  27.         injectView(this);  
  28.           
  29.         //直接使用UI元素  
  30.         mytext.setText("Text0");  
  31.         myimageview.setImageResource(R.drawable.junny);  
  32.     }  
  33.       
  34.     /** 实例化@InjectView 注解的成员*/  
  35.     public void injectView(Activity activity)  
  36.     {  
  37.         Field[] fields = activity.getClass().getDeclaredFields();//得到Activity中的所有定义的字段  
  38.         if (fields != null && fields.length > 0)  
  39.         {  
  40.             for (Field field : fields)  
  41.             {  
  42.                 if (field.isAnnotationPresent(InjectView.class))//方法返回true,如果指定类型的注解存在于此元素上  
  43.                 {  
  44.                     Log.i("Field", field.toString());  
  45.                       
  46.                     InjectView mInjectView = field.getAnnotation(InjectView.class); //获得该成员的annotation  
  47.                     int viewId = mInjectView.id();  //获得该注解的id  
  48.                     View view=activity.findViewById(viewId);//获得ID为viewID的组件对象  
  49.                       
  50.                     Log.i("Field", String.valueOf(viewId));  
  51.                     Log.i("Field", view.getClass().toString());  
  52.                       
  53.                     try  
  54.                     {  
  55.                         field.setAccessible(true);//设置类的私有成员变量可以被访问  
  56.                         field.set(activity, view);//field.set(object,value)===object.fieldValue = value  
  57.                     } catch (Exception e) { e.printStackTrace();}  
  58.                 }  
  59.                 else  
  60.                     Log.i("Field", "该字段没有被注解");  
  61.             }  
  62.         }  
  63.     }  
  64. }  

布局文件 activity_main.xml
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:background="#000000"  
  6.     android:orientation="vertical"  
  7.     tools:ignore="ContentDescription" >  
  8.     <ImageView  
  9.         android:id="@+id/myimgview"  
  10.         android:layout_width="wrap_content"  
  11.         android:layout_height="wrap_content"  
  12.         android:layout_gravity="center"/>  
  13.     <TextView  
  14.         android:id="@+id/mytext"  
  15.         android:layout_width="match_parent"  
  16.         android:layout_height="0dp"  
  17.         android:layout_weight="1"  
  18.         android:gravity="center"  
  19.         android:background="@android:color/darker_gray"  
  20.         android:textSize="30sp" />  
  21. </LinearLayout>  

实验结果

即注入成功了 大笑


需要说下我遇到的问题,在以上的基础上,如果将布局文件里的TextView 和ImageView两个布局换个位置,这时候再运行下,会出现空指针异常。

将Log向上翻会发现一个警告

即非法参数异常

定位到这一行

  1. field.set(activity, view);  

通过观察Log打印的日志

发现第一二行是对的,即获得注解的类型和ID(ImageView 2131230720)都是正确的,但是通过findViewById获取view的时候,Log第三行显示的却是TextView。

百思不得其解,最后无赖之下,清理一下项目,搞定。

我不知道是怎么回事,暂且推测为资源缓存吧。

posted on 2013-12-10 19:20 大龙 阅读(1600) 评论(0)  编辑 收藏 引用


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理