Java注解的作用

注解的好处:

1.能够读懂别人写的代码,特别是框架相关的代码。

2.本来可能需要很多配置文件,需要很多逻辑才能实现的内容,就可以使用一个或者多个注解来替代,这样就使得编程更加简洁,代码更加清晰。

3.(重点)刮目相看。
(但是怎么样才能让别人刮目相看呢?会用注解不是目的,最重要的是要使用自定义注解来解决问题。)
举个栗子:
如果面试的时候,你跟老板说你会使用注解,老板觉得你这个人还行;但是如果老板发现你会自定义注解解决问题,老板肯定就会眼前一亮。

注解这一概念是在java1.5版本提出的,说Java提供了一种原程序中的元素关联任何信息和任何元数据的途径的方法。

一、Java中的常见注解

1)JDK注解
JDK注解一共分为三类:

 

JDK注解.png

案例:
我们先新建一个接口people,如下:

1

2

3

4

5

public interface people {

    public String name();

    public int age();

    public void work();

}

然后再建一个类Child实现类people这个接口,并实现该类的方法:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

public class Child implements people {

    @Override

    public String name() {

        return null;

    }

 

    @Override

    public int age() {

        return 0;

    }

 

    @Override

    public void work() {

 

    }

看到这里,我们发现这里的所有方法都会加上一个@Override标记,它告诉我们,同时也告诉编译器我们的这些方法肯定覆盖了类people里面的方法的。假如说,我现在把类people里面的某一个方法注释掉:

1

//public String name();

再看类Child里面的name方法就会报错。这样,以后大家看到@Override的时候就能想到这个方法是覆盖了某个接口的方法的。

然后,我们回过头来看类people里面有一个work的方法。这里我们可以理解为人是要工作的,但是并不是所有的人都在工作,那么怎么办呢?如果说这个接口正在用,我们不能删除这个方法,这个时候我们就可以这样:

1

2

@Deprecated

public void work();

@Deprecated标记就表明这个方法已经过时了,在实际中,它又有什么样的应用场景呢?我们在建一个测试类:

1

2

3

4

5

6

public class Test {

    public void work() {

        people people=new Child();

!        people.work();

    }

}

这个时候我们会发现myeclipse会给一个警告,并且在work中间出现一个破折号,意思就是这个方法已经过时了。那么问题来了,虽然这个方法过时了,但是我们就是那么傲娇,一定要用它,怎么办呢?只需要这样:

1

2

3

4

5

6

7

public class Test {

    @SuppressWarnings("deprecation")

    public void work() {

        people people=new Child();

        people.work();

    }

}

这样我们就忽略了这个警告。@SuppressWarnings(“deprecation”)就表示我们忽略了deprecation这样的一个警告。

2)Java第三方注解

 

第三方注解.png

二、注解的分类

1)按照运行机制划分:
【源码注解→编译时注解→运行时注解】

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

编译时注解:在源码和.class文件中都存在。像前面的@Override、@Deprecated、@SuppressWarnings,他们都属于编译时注解。

运行时注解:在运行阶段还起作用,甚至会影响运行逻辑的注解。像@Autowired自动注入的这样一种注解就属于运行时注解,它会在程序运行的时候把你的成员变量自动的注入进来。

2)按照来源划分:
【来自JDK的注解——来自第三方的注解——自定义注解】

3)元注解:
元注解是给注解进行注解,可以理解为注解的注解就是元注解。

三、自定义注解

我们分四步来解析自定义注解:
自定义注解的语法要求:

1

2

3

4

5

6

7

8

9

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

@Retention(RetentionPolicy.RUNTIME)

@Inherited

@Documented

public @interface Description {

    String desc();

    String author();

    int age() default 18;

}

首先我们要明确这不是一个接口,它是使用@interface关键字定义的一个注解。
然后我们看下面的几个方法,String desc();虽然它很类似于接口里面的方法,其实它在注解里面只是一个成员变量(成员以无参无异常的方式声明),int age() default 18;(成员变量可以用default指定一个默认值的)。
最后我们要知道:
①.成员类型是受限制的,合法的类型包括基本的数据类型以及String,Class,Annotation,Enumeration等。
②.如果注解只有一个成员,则成员名必须取名为value(),在使用时可以忽略成员名和赋值号(=)。
③.注解类可以没有成员,没有成员的注解称为标识注解。

元注解:

有没有发现上面那段代码有一个没有说呢?没错,它们就是我们所说的元注解:

1

2

3

4

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

@Retention(RetentionPolicy.RUNTIME)

@Inherited

@Documented

我们先看第一行:@Target是这个注解的作用域,ElementType.METHOD是这个注解的作用域的列表,METHOD是方法声明,除此之外,还有:
CONSTRUCTOR(构造方法声明),FIELD(字段声明),LOCAL VARIABLE(局部变量声明),METHOD(方法声明),PACKAGE(包声明),PARAMETER(参数声明),TYPE(类接口)

第二行:@Retention是它的生命周期,前面不是说注解按照运行机制有一个分类嘛,RUNTIME就是在运行时存在,可以通过反射读取。除此之外,还有:
SOURCE(只在源码显示,编译时丢弃),CLASS(编译时记录到class中,运行时忽略),RUNTIME(运行时存在,可以通过反射读取)

第三行:@Inherited是一个标识性的元注解,它允许子注解继承它。

第四行:@Documented,生成javadoc时会包含注解。

使用自定义注解:
使用注解的语法:
@<注解名>(<成员名1>=<成员值1>,<成员名1>=<成员值1>,…)
案例:

1

2

3

4

@Description(desc="i am Color",author="boy",age=18)

public String Color() {

    return "red";

}

这里的Description是我们刚才在自定义注解语法要求里面定义的注解噢,然后我们可以给它的每一个成员变量赋值,注意数据类型。值得注意的是,因为我们前面定义的作用域是在方法和类接口上,所以这个注解在Color()方法上使用是没问题的。

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

准备工作:

 

Description类.png

 

Child类.png

接下来,我们就开始测试了:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

public class ParseAnn {

    public static void main(String[] args) {

        try {

            // 使用类加载器加载类

            Class c = Class.forName("com.test.Child");

            // 找到类上面的注解

            boolean isExist = c.isAnnotationPresent(Description.class);

            // 上面的这个方法是用这个类来判断这个类是否存在Description这样的一个注解

            if (isExist) {

                // 拿到注解实例,解析类上面的注解

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

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

            }

        } catch (ClassNotFoundException e) {

            e.printStackTrace();

        }

    }

}

输出的结果:
i am class annotation
可以看到,我们成功的解析了Child类上面的注解。

接下来,我们继续解析方法上的注解:

1

2

3

4

5

6

7

8

9

10

//获取所有的方法

Method[] ms = c.getMethods();

// 遍历所有的方法

for (Method m : ms) {

    boolean isExist1 = m.isAnnotationPresent(Description.class);

    if (isExist1) {

        Description d1=m.getAnnotation(Description.class);

        System.out.println(d1.value());

    }

}

输出的结果:
i am class annotation
i am method annotation
可以看到,我们成功的解析了方法上面的注解。

1

2

3

4

5

6

7

8

9

10

11

12

//另一种解析方法

for (Method m : ms) {

    //拿到方法上的所有的注解

    Annotation[] as=m.getAnnotations();

    for (Annotation a : as) {

        //用二元操作符判断a是否是Description的实例

        if (a instanceof Description) {

            Description d=(Description) a;

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

        }

    }

}

也可以得到上面的效果。

  • 12
    点赞
  • 52
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java注解是一种可插入的元数据,它可以提供给编译器、解释器和其他工具一些关于程序代码的信息。注解可以用于标记类、方法、字段、参数等元素,以便更好地理解它们的含义和用法。注解可以帮助开发人员更好地维护和扩展代码,例如用于自动生成文档、验证代码的正确性、标记测试用例等。此外,注解还可以与反射机制结合使用,使得程序可以在运行时获取注解的信息,从而实现更加灵活的程序设计。 ### 回答2: Java注解是一种在代码中添加元数据的方式,它可以用来给程序中的类、方法、字段等元素添加额外的信息。注解在编译时可以被读取,并且可以对代码的行为、结构和生成过程进行修改。 首先,注解可以提供编译时的静态检查。通过使用注解,我们可以在编译阶段对代码进行检查,以确保代码的正确性和一致性。例如,我们可以使用@Deprecated注解来标记被弃用的类、方法或字段,编译器在编译时会给出警告,以便开发者知道该部分代码已经不推荐使用。 其次,注解可以用来生成文档。注解可以提供额外的元数据,这些元数据可以被提取出来用于生成项目的文档。例如,我们可以使用注解来描述方法的参数、返回值和异常等信息,并通过工具生成详细的API文档。 另外,注解也可以用于运行时的动态处理。通过结合反射机制,我们可以在运行时读取和解析注解,并根据注解的信息对代码进行动态的处理。例如,Spring框架中的注解@Autowired可以自动注入依赖的实例,简化了代码编写和配置。 此外,注解也可以作为标记,用于标记特定类型或元素的特征或状态。例如,JUnit框架中的@Test注解可以标记测试方法,使得编写测试代码更加简单和直观。 总之,Java注解可以提供编译时的静态检查、生成文档、运行时动态处理和标记等多种作用。通过合理地使用注解,我们可以提高代码的可读性、简化开发流程、提高代码的可维护性。 ### 回答3: Java注解是一种用于提供元数据的标记机制,它可以用于代码的编译、运行和部署环节中。Java注解有以下几个主要的作用。 1. 提供编译时的类型检查:注解可以用于约束代码的使用,比如使用@Deprecated注解标记某个方法或类已经过时,这样在编译时就会发出警告,避免在使用时出现错误。 2. 提供运行时的功能扩展:注解可以在运行时通过反射机制获取到注解的信息,从而对代码进行相应的处理,实现各种功能扩展。比如JUnit的@Test注解可以标记测试用例方法,JUnit运行时会通过反射找到所有被@Test注解标记的方法,并执行相应的测试。 3. 自动生成代码:通过注解可以在编译时生成一些额外的代码,减少开发者的工作量。比如使用Lombok库中的@Data注解可以自动生成getter、setter等方法,使得代码编写更加简洁。 4. 生成文档:注解可以用于生成API文档,提供额外的文档信息。比如使用Javadoc注解可以将注解中的信息生成API文档,使得开发者可以更加方便地查阅和理解代码。 5. 使用注解框架:通过使用注解框架,可以快速实现各种功能,减少开发时间和复杂度。比如Spring框架中的注解可以用于自动注入依赖、事务管理等,简化了开发过程。 总的来说,Java注解可以提供编译和运行时的校验和功能扩展,使得代码更加灵活、简洁和易于维护。它是Java语言中的一个重要特性,被广泛应用于各个领域的开发中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值