zoukankan      html  css  js  c++  java
  • 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;
      }
    }
  • 相关阅读:
    Cayley's Tree Formula & Prufer's Method
    POJ 2262:Goldbach's Conjecture
    The Sieve of Eratosthenes (素数筛选法)
    POJ 2244:Eeny Meeny Moo(稍加变形的约瑟夫问题)
    POJ 1595:Prime Cuts
    iframe标签的使用
    js笔记
    Asp.Net知识点
    Reapte控件的使用
    浮躁十年
  • 原文地址:https://www.cnblogs.com/hupp/p/4705961.html
Copyright © 2011-2022 走看看