Java注解解析

概念

注解是Java提供的一种 源程序中的元素关联任何信息和任何元数据的途径和方法。

Java常见注解

  • JDK自带注解

    1. @Override 覆盖父类的方法

    2. @Deprecated 让方法过时

    3. @Suppvisewarnings 忽略警告

  • 常见第三方注解(Spring为例)

    1. @Autowired 可以对成员变量、方法和构造函数进行标注,来完成自动装配的工作。通过 @Autowired的使用来消除set ,get方法。

    2. @Service 用于标注业务层组件。定义某个类为一个bean,则在这个类的类名前一行使用@Service("XXX"),就相当于讲这个类定义为一个bean,bean名称为XXX。而无需去xml文件内去配置。

    3. @Repository 用于标注数据访问组件,即DAO组件。

注解分类

  • 按运行机制分

    1. 源码注解:注解只在源码中存在,编译成.class文件就不存在了

    2. 编译时注解:注解在源码和.class文件中都存在(如:JDK内置系统注解)

    3. 运行时注解:在运行阶段还起作用,甚至会影响运行逻辑的注解(如:Spring中@Autowried)

  • 按照来源分

    1. JDK内置注解

    2. 自定义注解

    3. 第三方注解

  • 元注解——注解的注解

自定义注解

  • 自定义注解的语法要求

    1. 使用@interface关键字定义注解

    2. 成员以无参无异常方式声明

    3. 可以用default为成员指定一个默认值

image.png

4. 如果注解只有一个成员,则成员名必须取名value(),在使用时可以忽略成员名和赋值

号(=);

5. 成员类型是受限的,合法的类型包括原始数据类型和String,Class,Annotation,Enumeration

6. 注解类可以没有成员,没有成员的注解称为标识注解

image.png
  • 元注解(注解里的注解)

    1. @Target({ElementType.FIELD,ElementType.METHOD}):定义注解的作用域

    • CONSTRUCTOR 构造方法声明

    • FIELD 字段声明

    • LOCAL_VARIABLE 局部变量声明

    • METHOD 方法声明

    • PACKAGE 包声明

    • PARAMETER 参数声明

    • TYPE 类接口

    2. @Retention(RetentionPolicy.RUNTIME):定义生命周期

    • SOURCE 只在源码显示,编译时会丢弃

    • CLASS 编译时会记录到class中,运行时忽略

    • RUNTIME 运行时存在,可以通过反射读取

    3. @Inherited:允许子类继承父类上的注解。只能用在类上,如果用在接口上将不会起作用。也只能继承类上的注解,方法上的不能被继承。

    4. @Documented:生成javadoc的时候包含注解

使用自定义注解


/* 自定义的注解类 */

import java.lang.annotation.Documented;

import java.lang.annotation.ElementType;

import java.lang.annotation.Inherited;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target({ElementType.METHOD, ElementType.FIELD, ElementType.CONSTRUCTOR})

@Retention(RetentionPolicy.RUNTIME)

@Inherited

@Documented

public @interface Description{

  int age() default 10;

  String author();

  String desc();

}

--------------------------------------------------------------------------------

/*

使用注解的语法:

@<注解名>(<成员名1>=<成员值1>,<成员名1>=<成员值1>,...)

*/

@Description(desc="I am eyeColor",author="Mooc boy",age=18)

public String eyeColor(){

  return "red";

}

解析注解

概念:通过反射获取类、函数或成员的运行时注解信息,从而实现动态控制程序运行的逻辑。

步骤

  • 使用类加载器加载类

  • 找到类上边的注解

  • 拿到注解实例

  • 遍历方法上的注解

示例


//自定义注解 Description.java

import java.lang.annotation.Documented;

import java.lang.annotation.ElementType;

import java.lang.annotation.Inherited;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target({ElementType.METHOD, ElementType.FIELD, ElementType.TYPE})

@Retention(RetentionPolicy.RUNTIME)

@Inherited

@Documented

public @interface Description{

  int age() default 10;

  String author();

  String desc();

}


// Person.class 用自定义注解修饰

@Description(desc="human class", author="wgp", age=20)

public class Person {

  @Description(desc="walk method", author="gp", age=21)

  public void walk(){

    System.out.println("human can walk!");

  }

}


// 注解解析类 Test.class

import java.lang.annotation.Annotation;

import java.lang.reflect.Method;

public class Test {

public static void main(String[] args) {

  try {

    //1\. 使用类加载器加载类

    Class c = Class.forName("Person");

    //2\. 找到类上面的注解

    boolean isExist=c.isAnnotationPresent(Description.class);//isAnnotationParse()判断类上是否存在Description这样的注解

    if(isExist){

      //3.获得注解实例

      Description d = (Description)c.getAnnotation(Description.class);

      System.out.println(d.desc());

    }

    //4.找到方法上的注解

    Method[] mts = c.getMethods();

    for (Method mt : mts) {

      if(mt.isAnnotationPresent(Description.class)){

        Description d2 = (Description)mt.getAnnotation(Description.class);

        System.out.println(d2.author());

      }

    }

    //另外一种获取类上的注解的途径

    Annotation[] annos = c.getAnnotations();

    for (Annotation anno : annos) {

      if(anno instanceof Description){

        System.out.println(((Description) anno).author());

      }

    }

    //另一种获取方法上的注解的途径

    for (Method mt : mts){

      Annotation[] annos2 = mt.getAnnotations();

      for (Annotation anno : annos2) {

        if(anno instanceof Description){

          System.out.println(((Description) anno).desc());

        }

      }

    }

  } catch (Exception e) {

    //TODO: handle exception

  }

}

}

项目案例

说明

代替Hibernate的解决方案,用在项目的持久层,核心代码就是用自定义注解实现的。

需求

1. 有一张用户表,字段包含id,用户名,密码,地址,手机号;

2. 方便的对每个字段或者每个字段的组合进行检索,并打印出SQL。

分析

1. 首先肯定有一个关于用户的JavaBean,包含所有用户信息;

2. 有了JavaBean需要将属性与字段匹配,也就是ORM;

3. 匹配完相当于拼接SQL语句时,知道表名和字段名称了,还缺少where中的字段值是多少;

4. 通过用户类的getter方法来获取字段值为多少,拼接完整SQL语句。

代码实现

1. 用户类,我们希望用@Table和@Cloumn来绑定;


/**

* User:用户JavaBean

*/

@Table("user")

public class User {

  @Column("id")

  public Integer id;

  @Column("user_name")

  public String userName;

  @Column("password")

  public String passWord;

  @Column("address")

  public String address;

  @Column("phone_num")

  public String phoneNum;

  public void setId(Integer id) {

    this.id = id;

  }

  public void setUserName(String userName) {

    this.userName = userName;

  }

  public void setPassWord(String passWord) {

    this.passWord = passWord;

  }

  public void setAddress(String address) {

    this.address = address;

  }

  public void setPhoneNum(String phoneNum) {

    this.phoneNum = phoneNum;

  }

  public Integer getId() {

    return id;

  }

  public String getUserName() {

    return userName;

  }

  public String getPassWord() {

    return passWord;

  }

  public String getAddress() {

    return address;

  }

  public String getPhoneNum() {

    return phoneNum;
  
  }

}

2. 自定义@Table注解和@Column注解;


import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

public @interface Table{

  String value();

}

--------------------------------------------------------------------------

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target(ElementType.FIELD)

@Retention(RetentionPolicy.RUNTIME)

public @interface Column{

  String value();

}

3. 根据反射来拼接出全部的SQL语句。


import java.lang.reflect.Field;

import java.lang.reflect.Method;

public class UserDao {

  public static void main(String[] args) {

    User user1 = new User();

    user1.setId(0);

    User user2 = new User();

    user2.setUserName("ping");

    User user3 = new User();

    user3.setAddress("beijing");

    user3.setPhoneNum("10101010101");

    String sql1 = query(user1); //查询id为0用户

    String sql2 = query(user2); //查询用户名含有ping的用户

    String sql3 = query(user3); //查询满足多个条件的用户

    System.out.println(sql1);

    System.out.println(sql2);

    System.out.println(sql3);
  }

  public static String query(User user) {

    StringBuilder sBuilder = new StringBuilder();

    Class clazz = user.getClass();

    // 获取表名

    if(!clazz.isAnnotationPresent(Table.class)){

      return null;

    }

    Table table = (Table) clazz.getAnnotation(Table.class);

    String tableName = table.value();

    sBuilder.append("select * from ").append(tableName).append(" where 1 = 1");

    // 获取所有字段名+值

    Field[] fields = clazz.getDeclaredFields();

    for (Field field : fields) {

      if(!field.isAnnotationPresent(Column.class)){

        continue;

      }

      Column column = field.getAnnotation(Column.class);

      String columnName = column.value(); //字段名

      String fieldName = field.getName(); //属性名

      String getMethodName = "get"+fieldName.substring(0, 1).toUpperCase()+fieldName.substring(1); //get方法名

      Object fieldValue = null; //属性值(先提升为Object类型)

      try {

        Method getMethod = clazz.getMethod(getMethodName);

        fieldValue = getMethod.invoke(user); //执行get方法获取值

      } catch (Exception e) {

        e.printStackTrace();

      }

      //继续拼接SQL

      if(fieldValue == null){

        continue;

      }

      sBuilder.append(" and ").append(columnName);

      if(fieldValue instanceof String){

        sBuilder.append("='").append(fieldValue).append("'");

      } else if(fieldValue instanceof Integer){

        sBuilder.append("=").append(fieldValue);

      }

    }

    return sBuilder.toString();
  }

}

参考资料

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,389评论 25 707
  • 霏霏秋雨点击着 满腹惆怅的梧桐树叶 幽静的小路悠长 我在遥望,遥望秋雨尽处 依旧秋雨茫茫 没有登上高山 你就不知道...
    泰安左眼皮跳跳阅读 202评论 0 5
  • 自己的世界难过的时候,外面的每一声都刺耳,每一秒都难熬。原以为时间可以过的很快是么?你看,煎熬的时候时间过的多慢,...
    周周大破坏之王阅读 118评论 0 0
  • 去江南找一匹白龙马 在绿林坐观一片叶子的飞翔 在这钟声里焦躁不安 使远方的云变得不安分起来 让人迷恋的风里 执一把...
    王艾柯阅读 211评论 1 1