Java 基础 - 注解

  • 注解是一种在原程序中的元素关联任何信息和任何元数据的途径和方法。
  • 在 Java 项目中、尤其是框架相关的代码,注解往往很常用,可以让代码更简洁清晰。

常见注解

JDK 自带注解

  • @Override:覆盖父类、实现接口的方法;
  • @Deprecated:方法已过时(可以被覆盖,但调用时会提示过时);
  • @SuppressWarnings("deprecation"):忽略方法过时的提示;
  • ...

第三方注解

Spring 注解:

  • @Autowired
  • @Service
  • @Repository
  • @Controller
  • ...

Mybatis 注解:

  • @InsertProvider
  • @UpdateProvider
  • @Options
  • ...
传统方法引入 DAO
使用注解引入 DAO

注解分类

按运行机制分类:

  • 源码注解:注解只在源码中存在,编译成 .class 文件就不存在了;
  • 编译时注解:在源码和 .class 文件中都存在;
  • 运行时注解:在运行阶段还起作用,甚至会影响运行逻辑。

按来源分类:

  • JDK 自带注解
  • 第三方(框架)注解
  • 自定义注解
  • 元注解

自定义注解

定义注解

// 元注解
@Target({ElementType.METHOD, ElementType.TYPE})     // 声明注解适用范围:构造方法、字段、局部变量、方法、包、参数、类和接口等
@Retention(RetentionPolicy.RUNTIME)                 // 生命周期:SOURCE(只存在于源码)、CLASS(记录到 class 中,运行时忽略)、RUNTIME(运行时存在,可反射)
@Inherited                                          // 在父类上添加了注解允许子类继承
@Documented                                         // 生成 JavaDoc 时包含注解信息
public @interface Description {     // @interface 关键字定义注解

    // 注解可以没有成员,没有成员的注解称为标识注解
    // 如注解只有一个成员则必须取名为 value(),使用时可以忽略成员名和复制号(=)

    String desc();                  // 注解成员以无参无异常方式声明

    String author();                // 成员类型受限:String、Class、Annotation、Enumeration

    int age() default 18;           // 可以为成员指定默认值
}

使用自定义注解

@Desciption(desc="my test", author="YWH", age=17)
public String test() {
    return "OK";
}

解析注解

通过反射获取类、方法或成员上的运行时注解信息(只能是运行时注解),从而实现动态控制程序运行的逻辑。

try {
    Class c = Class.forName("com.ywh.test.Test");
    boolean isExist = c.isAnnotationPresent(Description.class);    // 判断类上是否添加了注解
    if (isExist) {
        Description d = c.getAnnotation(Description.class);
        System.out.println(d.value());
    }
    Method[] ms = c.getMethods();
    for (Method m: ms) {
        boolean isMExist = m.isAnnotationPresent(Description.class);
        if (isMExist) {
            Description d = (Description) m.getAnnotation(Description.class);
            System.out.println(d.value());
        }
    }
    // 另一种解析方法
    for(Method m: ms) {
        Annotation[] as = m.getAnnotations();
        for (Annotation a: as) {
            if (a instanceof Description) {
                Description d = (Description) a;
                System.out.println(d.value());
            }
        }
    }
}
catch (ClassNotFoundException e) {
    e.printStackTrace();
}

实例:使用注解实现 ORM 框架(生成查询语句)

Table.java

// 表注解
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface Table {
    String value();
}

Column.java

// 字段注解
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Column {
    String value();
}

定义测试数据模型

@Table("user")
public class Filter {

    @Column("id")
    private int id;

    @Column("user_name")
    private String name;

    public int getId() { return id; }

    public void setId(int id) { this.id = id; }

    public String getName() { return name; }

    public void setName(String name) { this.name = name; }
}

其对应的 MySQL 表:

CREATE TABLE user (
    id INT,
    user_name VARCHAR(32)
);

Test.java

public class Test {

    public static void main(String[] args) {
        Filter f1 = new Filter();
        f1.setId(1);

        Filter f2 = new Filter();
        f2.setName("ywh");

        String sql1 = query(f1);
        String sql2 = query(f2);
        System.out.println(sql1);
        System.out.println(sql2);
    }


    private static String query(Filter f) {

        // 待拼装的 SQL 语句
        StringBuilder sb = new StringBuilder();

        // 1. 获取类类型,通过反射获取类注解、从 Table 注解中提取出对应的数据库表名
        Class c = f.getClass();
        boolean exists = c.isAnnotationPresent(Table.class);
        if (!exists) {
            return null;
        }
        Table t = (Table) c.getAnnotation(Table.class);
        String tableName = t.value();

        sb.append("SELECT * FROM ").append(tableName).append(" WHERE 1 = 1");


        // 2. 遍历数据模型类的所有字段(对应数据库表的字段),拼装查询条件
        Field[] fArray = c.getDeclaredFields();
        for (Field field : fArray) {

            // 2.1 获取字段名:从 Column 注解中获取字段对应的数据库字段名
            boolean fExists = field.isAnnotationPresent(Column.class);
            if (!fExists) {
                continue;
            }
            Column column = field.getAnnotation(Column.class);
            String columnName = column.value();

            // 2.2 获取字段值:根据类的字段名拼装出 getter 方法,求得字段值
            String fieldName = field.getName();
            String getMethodName = "get" + fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);
            Object fieldValue = null;
            try {
                Method getMethod = c.getMethod(getMethodName);
                fieldValue = getMethod.invoke(f);
            } catch (Exception e) {
                e.printStackTrace();
            }

            // 2.3 拼装 SQL(暂不处理空值和等于 0 的值)
            if (fieldValue == null ||
                    (fieldValue instanceof Integer && (Integer) fieldValue == 0)
            ) {
                continue;
            }
            sb.append(" AND ").append(columnName);

            // 字段值为 String 类型
            if (fieldValue instanceof String) {

                // 以逗号割开,则解析为 IN 条件
                if (((String) fieldValue).contains(",")) {
                    String[] values = ((String) fieldValue).split(",");
                    sb.append(" IN (");
                    for (String v : values) {
                        sb.append("'").append(v).append("'").append(",");
                    }
                    sb.deleteCharAt(sb.length() - 1);       // 删除最后一个逗号
                    sb.append(")");
                } else
                    sb.append(" = ").append("'").append(fieldValue).append("'");
            }
            // 字段值为其他类型
            else if (fieldValue instanceof Integer) {
                sb.append(" = ").append(fieldValue);
            }
        }

        return sb.toString();
    }

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

推荐阅读更多精彩内容

  • 夯实 Java 基础 - 注解 不知道大家有没有一种感觉,当你想要了解某个知识点的时候,就会发现好多技术类 APP...
    醒着的码者阅读 1,052评论 4 7
  • 注解基本概念 我们在开发当中经常看到一些注解,例如override,Deprected等,这些注解再常见不过了,但...
    zhonj阅读 1,067评论 0 0
  • 文章开头先引入一处图片。 这处图片引自老罗的博客。为了避免不必要的麻烦,首先声明我个人比较尊敬老罗的。至于为什么放...
    小乖心塞阅读 352评论 1 0
  • 回顾 3.构造方法的私有化 六、在方法内部调用方法 七、方法的递归调用 八、代码块 学习小结 八、代码块 1.普通...
    砾桫_Yvan阅读 175评论 0 1
  • 你一把疏狂醉,你一盏清樽酒。把落寞书写得淋漓尽致,把沧桑付予一声叹息。那么多的,难过,你不说。 从晋国公子享尽富贵...
    春余清歌阅读 1,409评论 15 19