从Xutils运行时注解复习Java注解

  • 最近在做复习整理,顺便把整理的一些东西记录下来。
  • Annotation(注解)是JDK1.5及以后版本引入的。它可以用于创建文档,跟踪代码中的依赖性,甚至执基本编译时检查。几乎所有的框架设计中都会用到注解类。
  • 先看一下一个注解类的格式

    @Target(ElementType.METHOD)
    @Retention(RetentionPolicy.RUNTIME)
    public @interface NoodlesClick {
        int value() default 1;
    }
    
    • Target 表示这个注解可以放在那里使用,ElementType枚举常用有 PACKAGE、TYPE、METHOD、FIELD 几种,分别表示可以用在 包,类,方法,和成员变量上面。我们最常见的注解@Override就是用在方法上的。
    • Retention 表示这个注解可以保留到什么时候。RetentionPolicy枚举有SOURCE、CLASS、RUNTIME分别表示,在保留在源.java文件,javac后的.class文件,和保留到程序运行时。这篇写的就是要保存在运行时的。
    • 注解的声明用@interface声明。

先创建注解类

今天只实现View的绑定和点击事件。

  • 先创建两个注解类

        //需要导入的包
        import java.lang.annotation.ElementType;
        import java.lang.annotation.Retention;
        import java.lang.annotation.RetentionPolicy;
        import java.lang.annotation.Target;
    
    • 这个是点击的注解类NoodlesClick 只能用在方法上。

      @Target(ElementType.METHOD)
      @Retention(RetentionPolicy.RUNTIME)
      public @interface NoodlesClick {
          int value() default 1;
      }
      
    • 这个是View绑定的注解类NoodlesInject 只能用在成员变量上。

      @Retention(RetentionPolicy.RUNTIME)
      @Target(ElementType.FIELD)
      public @interface NoodlesInject {
          int value() default 1;
      }
      
  • 下面是常规的布局文件 和MainActivity的创建

    • 布局文件,只有一个TextView和Button

    • 这是MainActivity的代码

      public class MainActivity extends Activity {
      
          /**
           * 这里引用注解 @NoodlesInject
           * 后面括号里面的value 为 刚才注解里面的方法的返回值,方法默认值为1;
           *  int value() default 1;
           *  这里将TextView的id传进去。
           */
          @NoodlesInject(value = R.id.tv)
          TextView mTv;
      
          @Override
          protected void onCreate(Bundle savedInstanceState) {
              super.onCreate(savedInstanceState);
              setContentView(R.layout.activity_main);
              /**
               * 这里是重点。将当前对象传进去,进行View的绑定以及事件的绑定
               */
              NoodlesUtil.inject(this);
          }
      
          /**
           * 这里引用注解 @NoodlesClick
           * 后面括号里面的value 为 刚才注解里面的方法的返回值,方法默认值为1;
           *  int value() default 1;
           *  这里将Button的id传进去。
           */
          @NoodlesClick(value = R.id.btn)
          private void onClick(View v) {
              Toast.makeText(this, mTv.getText().toString().trim(), Toast.LENGTH_SHORT).show();
          }
      }
      
    • NoodlesUtil是用来做具体的注解与Activity的操作的。

      public class NoodlesUtil {
      
          /**
           * 
           * DESC : 静态方法,用于给activity层调用 . 
      * @param activity 要解析此activity中的成员变量和方法。 */ public static void inject(Activity activity) { bindView(activity); bindClick(activity); } /** * * DESC : 解析成员变量 .
      * @param activity */ private static void bindClick(Activity activity) { try { //获取activity的class。 Class clazz = activity.getClass(); //利用反射 获取activity中的所有成员变量 Field[] fields = clazz.getDeclaredFields(); //遍历成员变量 for (Field field : fields) { /** * getAnnotation 方法为获取成员变量field上是否有NoodlesInject注解。 * @NoodlesInject(value = R.id.tv) * TextView mTv; * 这是Activity中写的成员变量,如果返回不为null,则表明上面有此注解 */ NoodlesInject annotation = field.getAnnotation(NoodlesInject.class); if (annotation != null) { //调用注解中的方法,value 获取到值 R.id.tv int id = annotation.value(); //通过id找View,将本来需要在activity中写的代码放在了这里来。 View view = activity.findViewById(id); field.setAccessible(true); //将找到的View赋值给 textView 类似于 TextView mTv = (TextView)view; field.set(activity, view); } } } catch (Exception e) { e.printStackTrace(); } } /** * * DESC : 解析绑定了点击事件的注解类 .
      * @param activity */ private static void bindView(final Activity activity) { try { //获取activity的class。 Class clazz = activity.getClass(); //利用反射 获取activity中的所有的方法 Method[] methods = clazz.getDeclaredMethods(); //遍历方法 for (final Method method : methods) { /** * getAnnotation 方法为获取方法method上是否有NoodlesClick注解。 * @NoodlesClick(value = R.id.btn) * private void onClick(View v) { ... } * 这是Activity中写的方法,如果返回不为null,则表明上面有此注解 */ NoodlesClick annotation = method.getAnnotation(NoodlesClick.class); if (annotation != null) { //调用注解中的方法,value 获取到值 R.id.btn int id = annotation.value(); //通过id找View。 final View view = activity.findViewById(id); //设置view的点击事件。 view.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { try { method.setAccessible(true); //将View传入method方法,并执行method方法 // --》 就是执行Activity中的onclick方法。 method.invoke(activity, view); } catch (Exception e) { e.printStackTrace(); } } }); } } } catch (Exception e) { e.printStackTrace(); } } }

      具体的执行,已经在代码的注释中,这个demo中除了布局文件之外的代码已经全部贴在这里了,运行时注意导包。
      xutils3.0 和butterknife 现在已经全部将运行时改为编译时了,至于编译时的原理,后面会整理出来。

你可能感兴趣的:(注解)