Android开源框架-Annotation框架(以ViewMapping注解为例)

Annotation 分类

1 标准 Annotation

包括Override, Deprecated, SuppressWarnings,标准 Annotation 是指 Java 自带的几个 Annotation,上面三个分别表示重写函数,函数已经被禁止使用,忽略某项 Warning

2 元 Annotation

@Retention, @Target, @Inherited, @Documented,元 Annotation 是指用来定义 Annotation 的 Annotation,在后面 Annotation 自定义部分会详细介绍含义

3 自定义 Annotation

自定义 Annotation 表示自己根据需要定义的 Annotation,定义时需要用到上面的元 Annotation 这里只是一种分类而已,也可以根据作用域分为源码时、编译时、运行时 Annotation,后面在自定义 Annotation 时会具体介绍

1.Java.lang包中常用的注解有@Override,@Deprected(已经废弃),@SupressWarning(屏蔽掉一些警告。)我们可以自定义注解。

2.Java注解之@Retention,@Documented,@Inherited.

  • Retention注解,保留注解说明,这种类型的猪儿会被保留到哪个阶段,有三个值:RetentionPolicy.SOURCE(只有源码级别保留,编译时被忽略),RetentionPolicy.CLASS(class文件中保留,Jvm被忽略),RetentionPolicy.RUNTIME(Jvm保留,所以他们能在运行时被Jvm或者其他运用反射机制的代码所读取和使用)。
  • Document注解,目的就是将这一Annotation的信息显示在JavaAPI文档上,如果吗没有这个注解,在API文档上就不显示Annotation信息。这个注解应该被Javadoc工具记录,javadoc工具在默认情况下不包括注解,如果被注解,会被javadoc之类的工具处理,注解类型的信息也会被包括在生成的文档中。
  • Inherited注解,子类的父类被标记为@Inherited,子类将自动继承父类的所有属性,除了要实现父类接口方法,还要事项Object类中的方法及annotationType()。

3.java中注解主要的功能是:让开发更易,开发和部署工具可以以某种方式阅读并处理他们。包含注解的程序可以替代额外的Java源文件,XML文件或错误检测机制。

4.Java中注解的目的是:花更少得 时间在哪些死硬无用的细节中,更多的关注业务逻辑。利用注解可以明显的减少代码量,使得代码的结构清晰。

5.Java中的反射机制:是Java被视为动态(或者准动态)语言的一个关键性质,这个机制允许程序在运行是通过jdk给出的reflectionAPIs获取任何一个已知名称的class内部信息,包括其modifier(诸如:public,static等),superclass(例如:object),实现interface(例如:cloneable),也包括fileds和methods所有信息,并可以运行改变fields和methods。Java可以加载一个运行时才得知名称的class,获取其完整的结构。

6.反射机制的功能:获取类的class对象,获取类的fields,获取类的method,获取类的constructor,新建类的实例函数(class<T>的函数newInstance)。

7.Target里面的ElementType是用来指定Annotation类型可以用在哪些元素上的。例如:TYPE(类型),FIELD(属性),METHOD(方法),PARAMETER(参数),CONSTRUCTOR(构造函数),LOCAL_VARIABLE(局部变量),PACKAGE(包)。其中TYPE类型是指可以用在class,interface,enum,Annotatin类型上。

下面我们来看一下Android中的ViewMapping注解:

从上图我们可以看到,在这个注解上面有定义该注解的保留阶段是在Jvm。可以用在类型和属性上面。

其中定义部分的id()(方法)表示注解接收一个int类型的数据作为id所对应的值(id=R.id.xx)。注解类型不能new对象,viewMapping对象从何而来,Java反射机制,每个继承object类都有getClass方法。通过获取注解类的class,再获取其中的id,然后就可以获取id对应的控件。

Android客户端用注解来实现findViewById功能及setContentView功能,在Androidplus.jar中的util中有如下代码实现。

package com.androidplus.util;

import android.app.Activity;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import java.lang.reflect.Field;

public class ViewMappingUtil
{
  public static View mapView(Object object, Activity activity)
  {
    Context context = activity;
    ViewMapping viewMapping = (ViewMapping)object.getClass().getAnnotation(ViewMapping.class);
    View rootView = ((LayoutInflater)context.getSystemService("layout_inflater")).inflate(viewMapping.id(), null);
    activity.setContentView(rootView);
    mapView(object, rootView);
    return rootView;
  }

  public static void mapView(Object object, View rootView) {
    startReflect(object, rootView, object.getClass());
  }

  public static void mapView(Object object, View rootView, int layer) {
    startReflect(object, rootView, object.getClass());
    Class clazz = object.getClass().getSuperclass();
    for (int i = 0; i < layer - 1; ) {
      startReflect(object, rootView, clazz);

      ++i; clazz = clazz.getSuperclass();
    }
  }

  private static void startReflect(Object object, View rootView, Class<?> clazz)
  {
    Field[] fields = clazz.getDeclaredFields();

    Field[] arr$ = fields; int len$ = arr$.length; for (int i$ = 0; i$ < len$; ++i$) { Field field = arr$[i$];
      ViewMapping viewMapping = (ViewMapping)field.getAnnotation(ViewMapping.class);
      int id = 0;
      if (viewMapping != null)
        try {
          id = viewMapping.id();
          field.setAccessible(true);
          field.set(object, rootView.findViewById(id));
        } catch (Exception e) {
          throw new RuntimeException(id + " map error!");
        }
    }
  }

  public static View mapView(Object object, Context context)
  {
    return mapView(object, context, null, false);
  }

  public static View mapView(Object object, Context context, ViewGroup parent, boolean isAttach) {
    ViewMapping viewMapping = (ViewMapping)object.getClass().getAnnotation(ViewMapping.class);
    View rootView = ((LayoutInflater)context.getSystemService("layout_inflater")).inflate(viewMapping.id(), parent, isAttach);

    mapView(object, rootView);
    return rootView;
  }
}

 

posted @ 2015-08-05 21:56  lisahappy  阅读(591)  评论(0编辑  收藏  举报