Android-注解

java注解在 Android 中有两种应用方式,一种方式是基于反射的,在程序的运行期间获取类信息进行反射调用;另一种是使用注解处理,在编译期间生成相关代码,然后在运行期间通过调用这些代码来实现相关功能。 我们先了解一下注解的分类及其关键字

标准注解(java默认提供)

元注解(用户自己定义用)

@Override

@Target

@Deprecated

@Retention

@SuppressWarnnings

@Documented

@SafeVarags

@Inherited

@Repeatable

由表格可以看出java中的注解主要分为两类,标准注解和元注解,。标准注解是 Java 为我们提供的预定义注解,这个我们没多大关系,主要是元注解,元注解是用来提供给用户自定义注解用的,接下来我们来学习一下元注解。

JAVA 元注解

先解释每个注解的含义:

1.元注解之Target

@Target: 注解的作用:

  • @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(ElementTypeTYPE_PARAMETER) //类型参数声明
  • @Target(ElementType.TYPE_USE) //使用类型

2.元注解之Retention

@Retention:注解的保留位置

  • @Retention(RetentionPolicy.SOURCE) //注解仅存在于源码中,在class字节码文件中不包含,注解将被编译器丢弃.
  • @Retention(RetentionPolicy.CLASS) // 默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得,会被JVM丢弃.
  • @Retention(RetentionPolicy.RUNTIME) // 注解会在class字节码文件中存在,在运行时可以通过反射获取到

3.元注解之Document

  • @Document:说明该注解将被包含在javadoc中

4.元注解之Inherited

  • @Inherited:说明子类可以继承父类中的该注解

5.元注解之Repeatable

  • @Repeatable:在需要对同一种注解多次使用应用于指定对象,往往需要借助@Repeatable,为java8新增。

了解完这些注解的含义,我们来自定义一个,java注解有两种实现方式。

JAVA自定义注解

1.基于反射使用注解

首先我们的目标是制作一个用户信息表,是一个User对象,表中有其对应的属性,将注解和属性等关联,然后再通过反射拿到对应的注解值和属性值打印,思路如下: 1.1. 首先定义两个注解UserAnnotation和UserAttribute,创建一个注解遵循: public @interface 注解名 {方法参数} 1.2. 将注解和User对象关联给用户赋值。 1.3. 获取注解,打印注解的注解值和属性值。 定义注解代码如下

image.png

将注解和User对象绑定

image.png

主要通过获取注解打印

 public static void main(String [] args){
      UserBean mUserBean =new UserBean();
        mUserBean.setUserName("张三");
        mUserBean.setUserSex("男");
        mUserBean.setUserAge(18);
      StringBuffer mStringBuffer= printData(mUserBean);
      System.out.println(mStringBuffer);

    }

我们主要看一下printData这个打印的代码,都有注释不解释

    private static StringBuffer printData(UserBean userBean){
        //创建一个StringBuffer对象拼接数据
        StringBuffer mStringBuffer =new StringBuffer();

        //根据对象获取注解Class
        Class annotationClass =userBean.getClass();
        //判断对象中有没有我们定义的userAnnotation注解
        boolean bUserAnnotation =annotationClass.isAnnotationPresent(CustomeAnnotation.UserAnnotation.class);
        if(bUserAnnotation){
            //获取userAnnotation注解
            CustomeAnnotation.UserAnnotation mUserAnnotation= (CustomeAnnotation.UserAnnotation) annotationClass.getAnnotation(CustomeAnnotation.UserAnnotation.class);
            //获取获取userAnnotation注解的值
            String userAnnotationName =mUserAnnotation.userClassName();
            mStringBuffer.append(userAnnotationName+"信息如下:");

            //进而获取对象中的所有属性
            //获取对象中的所有属性
            Field[] fields = annotationClass.getDeclaredFields();
            //便遍历属性
             for(Field field : fields){
                 //判断属性中是否有我们定义的UserAttribute
                 boolean bUserAttribute =field.isAnnotationPresent(CustomeAnnotation.UserAttribute.class);
                 if(bUserAttribute){
                     //获取自定义UserAttribute注解
                    CustomeAnnotation.UserAttribute userAttribute = field.getAnnotation(CustomeAnnotation.UserAttribute.class);
                     //获取去userAttribute中的值
                     String name = userAttribute.userAttribute();
                     Object value = "";
                     try {
                         //获取对应属性的值,toUpperCase()把字符串转换为大写
                         Method method = annotationClass.getMethod("get" + field.getName().substring(0, 1).toUpperCase() + field.getName().substring(1));
                         value = method.invoke(userBean);
                     } catch (Exception e) {
                         e.printStackTrace();
                     }
                     //string类型
                     if (value instanceof String) {
                         mStringBuffer.append(name + "=").append(value).append(",");
                     } else if (value instanceof Integer) {
                         mStringBuffer.append(name + "=").append(value).append(",");
                     }

                 }else{
                     throw new NullPointerException("UserAttribute not find");
                 }
             }

        }else{//如果不存在抛一个异常
            throw new NullPointerException("userAnnotation not find");
        }
        return mStringBuffer;
    }

这样的话我们就可以成功打印出信息了,请问明白了没,如果没有明白,没关系我们继续在写一个,类似Butterknife的:

  • 第一步定义注解
  • 第二步将注解应用于控件
  • 在App运行时,通过反射将findViewbyId得到的控件,注入到注解描述的成员变量中完成绑定。 第一步:

image.png 第二步:

image.png

第三步:

image.png

2. 基于 annotationProcessor 使用注解

这个看到过一篇文章比较专业,我就不再这里说了

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏懒人开发

Eventbus3代码分析(二):注解入门

我们扯淡到java的知识,最好是先参考别人的文章,或者看官方的 自己能力有限,所以只是简单扯扯,有兴趣,最好自己看官方的解释

7010
来自专栏向治洪

java注解及在butternife中的实践和原理

1. 背景 之前去一个公司,说到了java的注解,问java的注解有几种方式,然后我提到了android中的butternife和afinal注解工具,我们知道...

22750
来自专栏菩提树下的杨过

mybatis的物理分页:mybatis-paginator

文件分割与合并是一个常见需求,比如:上传大文件时,可以先分割成小块,传到服务器后,再进行合并。很多高大上的分布式文件系统(比如:google的GFS、taoba...

23690
来自专栏拭心的安卓进阶之路

Java 进阶巩固:什么是注解以及运行时注解的使用

这篇文章 2016年12月13日星期二 就写完了,当时想着等写完另外一篇关于自定义注解的一起发。结果没想到这一等就是半年多 - -。 有时候的确是这样啊,总想着...

22160
来自专栏偏前端工程师的驿站

Java魔法堂:自定义和解析注解

一、前言                               注解(Annotation)作为元数据的载体,为程序代码本身提供额外的信息,使用过MyBa...

197100
来自专栏跟着阿笨一起玩NET

c#常用工具类:文件和二进制转换

18820
来自专栏pangguoming

JDBC上关于数据库中多表操作一对多关系和多对多关系的实现方法

我们知道,在设计一个Java bean的时候,要把这些BEAN 的数据存放在数据库中的表结构,然而这些数据库中的表直接又有些特殊的关系,例如员工与部门直接有一对...

88170
来自专栏Java技术栈

JAVA元注解@interface详解(@Target,@Documented,@Retention,@Inherited)。

jdk1.5起开始提供了4个元注解,用来定义自定义注解的注解,它们分别是: @Target 指定注解使用的目标范围(类、方法、字段等),其参考值见类的定义:ja...

37880
来自专栏吴伟祥

初识ByteBuffer 原

ByteBuffer 是java nio 和aio 编程所必须掌握的一个数据结构,也是掌握tio 所必须要学会的基础知 识。

5910
来自专栏我是攻城师

高效读取大数据文本文件(上亿行数据)

53740

扫码关注云+社区

领取腾讯云代金券