文档章节

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

JayPark不作死
 JayPark不作死
发布于 2014/04/09 13:29
字数 899
阅读 84
收藏 0

/**************************************************************************************************

 *  本博客为CSDN博主【MK】原创,博客地址:http://blog.csdn.net/mkrcpp/article/details/13509051

 **********************************************************************************************************************/

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

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

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

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

  • 首先注入布局资源(绑定layout布局)

  • 其次注入成员资源(绑定组件资源)

  • 然后注入成员变量(初始化普通变量)

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

我写了个小demo,MyAnno

InjectView.java

package com.myanno;  
  
import java.lang.annotation.ElementType;  
import java.lang.annotation.Retention;  
import java.lang.annotation.RetentionPolicy;  
import java.lang.annotation.Target;  
  
  
/** 
 * 自定义注解,获取资源ID 
 * @使用方法 
 *              @InjectView(id = R.id.settingIv) 
 *              private ImageView imgSetting;  
 *  
 * @author      michael.mao@sosino.com 
 * @date        2013-10-29 
 * @description @Retention: 定义注解的保留策略 
 *              @Retention(RetentionPolicy.SOURCE)  //注解仅存在于源码中,在class字节码文件中不包含 
 *              @Retention(RetentionPolicy.CLASS)   //默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得, 
 *              @Retention(RetentionPolicy.RUNTIME) //注解会在class字节码文件中存在,在运行时可以通过反射获取到 
 *              @Inherited                          //说明子类可以继承父类中的该注解 
 * 
 *              @Target(ElementType.TYPE)           //接口、类、枚举、注解 
 *              @Target(ElementType.FIELD)          //字段、枚举的常量 
 *              @Target(ElementType.METHOD)         //方法 
 *              @Target(ElementType.PARAMETER)      //方法参数 
 *              @Target(ElementType.CONSTRUCTOR)    //构造函数 
 *              @Target(ElementType.LOCAL_VARIABLE) //局部变量 
 *              @Target(ElementType.ANNOTATION_TYPE)//注解 
 *              @Target(ElementType.PACKAGE)        //包    
 */  
  
@Target(ElementType.FIELD)  
@Retention(RetentionPolicy.RUNTIME)  
public @interface InjectView  
{  
    /** View的ID */  
    public int id() default -1;  
}


MainActivity.java

  1. package com.myanno;  
      
    import java.lang.reflect.Field;  
      
    import android.app.Activity;  
    import android.os.Bundle;  
    import android.util.Log;  
    import android.view.View;  
    import android.widget.ImageView;  
    import android.widget.TextView;  
      
    public class MainActivity extends Activity {  
      
        /** 注解绑定UI元素 */  
        @InjectView(id=R.id.myimgview)  
        ImageView myimageview;  
          
        @InjectView(id=R.id.mytext)  
        TextView mytext;  
      
        @Override  
        protected void onCreate(Bundle savedInstanceState) {  
            super.onCreate(savedInstanceState);  
            setContentView(R.layout.activity_main);  
              
            //初始化注解绑定的成员变量  
            injectView(this);  
              
            //直接使用UI元素  
            mytext.setText("Text0");  
            myimageview.setImageResource(R.drawable.junny);  
        }  
          
        /** 实例化@InjectView 注解的成员*/  
        public void injectView(Activity activity)  
        {  
            Field[] fields = activity.getClass().getDeclaredFields();//得到Activity中的所有定义的字段  
            if (fields != null && fields.length > 0)  
            {  
                for (Field field : fields)  
                {  
                    if (field.isAnnotationPresent(InjectView.class))//方法返回true,如果指定类型的注解存在于此元素上  
                    {  
                        Log.i("Field", field.toString());  
                          
                        InjectView mInjectView = field.getAnnotation(InjectView.class); //获得该成员的annotation  
                        int viewId = mInjectView.id();  //获得该注解的id  
                        View view=activity.findViewById(viewId);//获得ID为viewID的组件对象  
                          
                        Log.i("Field", String.valueOf(viewId));  
                        Log.i("Field", view.getClass().toString());  
                          
                        try  
                        {  
                            field.setAccessible(true);//设置类的私有成员变量可以被访问  
                            field.set(activity, view);//field.set(object,value)===object.fieldValue = value  
                        } catch (Exception e) { e.printStackTrace();}  
                    }  
                    else  
                        Log.i("Field", "该字段没有被注解");  
                }  
            }  
        }  
    }


布局文件 activity_main.xml

  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
        xmlns:tools="http://schemas.android.com/tools"  
        android:layout_width="match_parent"  
        android:layout_height="match_parent"  
        android:background="#000000"  
        android:orientation="vertical"  
        tools:ignore="ContentDescription" >  
        <ImageView  
            android:id="@+id/myimgview"  
            android:layout_width="wrap_content"  
            android:layout_height="wrap_content"  
            android:layout_gravity="center"/>  
        <TextView  
            android:id="@+id/mytext"  
            android:layout_width="match_parent"  
            android:layout_height="0dp"  
            android:layout_weight="1"  
            android:gravity="center"  
            android:background="@android:color/darker_gray"  
            android:textSize="30sp" />  
    </LinearLayout>


实验结果


即注入成功了 大笑


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

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

即非法参数异常

定位到这一行

  1. field.set(activity, view);  


通过观察Log打印的日志

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

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

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

/**************************************************************************************************

 *  本博客为CSDN博主【MK】原创,博客地址:http://blog.csdn.net/mkrcpp/article/details/13509051

 **********************************************************************************************************************/


本文转载自:http://blog.csdn.net/mkrcpp/article/details/13509051

JayPark不作死
粉丝 15
博文 51
码字总数 13622
作品 0
苏州
程序员
私信 提问
阅读#ThinkAndroid#笔记(一)

今天看了ThinkAndroid源码的MVC实现部分,觉得存在不少问题。 一、设计TAActivity超类不合理 ThinkActivity框架要求所有的Activity组件继承TAActivity类,而TAActivity类是直接继承Activity类...

Callen
2015/08/08
0
0
移动开发者必须知道的Android框架推荐

一些总结出来的Android快速开发框架,全部都是开源框架,附带项目地址,是开发学习的绝佳资料。 thinkAndroid项目 github地址:https://github.com/white-cat/ThinkAndroid 功 能:ThinkAndr...

程序袁_绪龙
2014/09/02
0
0
【转】值得推荐的android开发框架简介

第一个:Afinal 项目地址:https://github.com/yangfuhai/afinal 功能:一个android的ioc,orm框架,内置了四大模块功 能:FinalAcitivity,FinalBitmap,FinalDb,FinalHttp。通过finalActivi...

kymjs张涛
2014/08/05
0
0
六款值得推荐的android(安卓)开源框架简介【转】

1、volley 项目地址 https://github.com/smanikandan14/Volley-demo (1) JSON,图像等的异步下载; (2) 网络请求的排序(scheduling) (3) 网络请求的优先级处理 (4) 缓存 (5) 多级别取消请求...

hkstar35
2014/07/11
0
0
六款值得推荐的android(安卓)开源框架简介【转】

1、volley 项目地址 https://github.com/smanikandan14/Volley-demo (1) JSON,图像等的异步下载; (2) 网络请求的排序(scheduling) (3) 网络请求的优先级处理 (4) 缓存 (5) 多级别取消请求...

火蚁
2014/07/09
0
1

没有更多内容

加载失败,请刷新页面

加载更多

阿里P8架构师谈:如何打造一份高并发编程知识体系

1.问题 1、什么是线程的交互方式? 2、如何区分线程的同步/异步,阻塞/非阻塞? 3、什么是线程安全,如何做到线程安全? 4、如何区分并发模型? 5、何谓响应式编程? 6、操作系统如何调度多线...

小刀爱编程
6分钟前
0
0
比特币:如何用地址查询交易?

在比特币应用开发中,一个常见的问题就是,在知道比特币地址的情况下,如何查询这个地址上发生的所有交易?或者类似的说法,如何查询一个指定的比特币地址发生的所有交易? 本文将给出这一问...

汇智网教程
9分钟前
0
0
Spring Boot 下,敏感词及特殊字符过滤处理方式

背景: 技术采用的是 Spring Boot ,请求方法主要为 POST, 请求使用较多的注解为 @RequestBody 交付测试人员进行测试,测试人员在对模糊搜索模块进行了各种特殊字符的搜索,以至于敏感词和特...

Ryan-瑞恩
14分钟前
0
0
使用 Jenkins X 渐进式交付

本文首发于:Jenkins 中文社区 这是渐进式交付系列的第二篇文章,第一篇请看:Kubernetes 中的渐进式交付:蓝绿部署和金丝雀部署。 我使用的我的 Croc Hunter 示例项目评估了 Jenkins X 中金...

Jenkins中文社区
21分钟前
1
0
零基础学算法->PI

本文章是介绍几种计算PI的方法 1.概率法计算PI(又称蒙特卡罗法) 1.1 定义,过程 在半径为1的圆1/4的区域,通过随机函数产生横纵坐标值x,y;当x*x+y*y<=1时,满足条件。 1.2 结果: 因为是随机...

tedzheng
24分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部