我们在使用Spring框架的时候,会常常使用类似:@Autowired 这种注解。

我们也能够自定义一些注解。Java的注解主要在包:java.lang.annotation中实现。

1. 元注解

什么是元注解?你能够这样理解。元注解是自己定义注解的注解。

元注解主要包括4个。

他们主要在java.lang.annotation中能够找到。

我们自己要创建注解的时候必需要用到这些元注解。

所以必须彻底理解这四个元注解的含义。

1. @Documented

2. @Inherited

3. @Retention

4. @Target

比如:

package com.test.www;

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; /**
* 定义一个username的自己定义注解
* @author zhuli
* @date 2014-7-5
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.TYPE, ElementType.METHOD})
@Inherited
public @interface UserNameAnnotations { public String value() default ""; }

1. @Documented

@Documented用于描写叙述其他类型的annotation应该被作为被标注的程序成员的公共API,因此能够被比如javadoc此类的工具文档化。

Documented是一个标记注解,没有成员。

2. @Inherited

@Inherited 元注解是一个标记注解。@Inherited阐述了某个被标注的类型是被继承的。

假设一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。

3. @Target

@Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法參数和本地变量(如循环变量、catch參数)。

ElementType.CONSTRUCTOR  作用于构造器

ElementType.FIELD  作用于域/属性

ElementType.LOCAL_VARIABLE  用于描写叙述局部变量

ElementType.METHOD  作用于方法

ElementType.PACKAGE   用于描写叙述包

ElementType.PARAMETER   用于描写叙述參数

ElementType.TYPE   用于描写叙述类、接口(包含注解类型) 或enum声明,最经常使用

单个修饰对象的范围:

@Target(ElementType.TYPE)

多个:

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

4. Retention

定义了该Annotation被保留的时间长短:某些Annotation仅出如今源码中,而被编译器丢弃。而还有一些却被编译在class文件里;编译在class文件里的Annotation可能会被虚拟机忽略。而还有一些在class被装载时将被读取(请注意并不影响class的运行,由于Annotation与class在使用上是被分离的)。使用这个meta-Annotation能够对 Annotation的“生命周期”限制。

RetentionPolicy.RUNTIME 注解会在class字节码文件里存在,在执行时能够通过反射获取到

RetentionPolicy.CLASS 默认的保留策略。注解会在class字节码文件里存在,但执行时无法获得

RetentionPolicy.SOURCE 注解仅存在于源代码中,在class字节码文件里不包括

2. 创建一个自己定义注解 - 作用于类

1. 创建一个注解类

package com.test.www;

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; /**
* 定义一个username的自己定义注解
* @author zhuli
* @date 2014-7-5
*/
@Documented //文档
@Retention(RetentionPolicy.RUNTIME) //在执行时能够获取
@Target({ ElementType.TYPE, ElementType.METHOD}) //作用到类,方法,接口上等
@Inherited //子类会继承
public @interface UserNameAnnotations { public String value() default ""; //使用的时候 @UserNameAnnotations(value="xxx") }

2. 创建一个Test类

package com.test.www;

/**
* 一个注解的測试类
* @author zhuli
* @date 2014-7-5
*/
//注入注解作用于类上面
//能够通过反射 获取类的信息之后 获取得到这个注解的值
@UserNameAnnotations(value = "initphp")
public class Test { private String userName; public String getUserName() {
return userName;
} public void setUserName(String userName) {
this.userName = userName;
} }

3. 測试类

package com.test.www;

public class mainTest {

    public static void main(String[] args) {
Class<Test> testClass = Test.class;
//由于注解是作用于类上面的,所以能够通过isAnnotationPresent来推断是否是一个
//有UserNameAnnotations注解的类
if (testClass.isAnnotationPresent(UserNameAnnotations.class)) {
System.out.println("this is a Annotations class");
//通过getAnnotation能够获取注解对象
UserNameAnnotations userNameAnnotations = (UserNameAnnotations) testClass.
getAnnotation(UserNameAnnotations.class);
if (userNameAnnotations != null) {
System.out.println("value:" + userNameAnnotations.value());
} else {
System.out.println("null");
}
} else {
System.out.println("this is not Annotations class");
} }
}

4. 输出:

this is a Annotations class
value:initphp

3. 创建一个自己定义注解 - 作用于方法

1. 自己定义注解类

package com.test.www;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* 定义一个作用到方法的注解
* @author zhuli.zhul
* @date 2014-7-5
*/
@Documented//文档
@Retention(RetentionPolicy.RUNTIME)//在执行时能够获取
@Target({ ElementType.TYPE, ElementType.METHOD })//作用到类,方法,接口上等
public @interface MethodType { //枚举类型
public enum MethodTypeEnum {
TYPE1, TYPE2
} //实际的值
public MethodTypeEnum methodType() default MethodTypeEnum.TYPE1;
}

2. 创建一个使用注解的类

package com.test.www;

import com.test.www.MethodType.MethodTypeEnum;

/**
* 一个注解的測试类
* @author zhuli
* @date 2014-7-5
*/
//注入注解作用于类上面
//能够通过反射 获取类的信息之后 获取得到这个注解的值
@UserNameAnnotations(value = "initphp")
public class Test { private String userName; //注解到
@MethodType(methodType=MethodTypeEnum.TYPE2)
public String getUserName() {
return userName;
} public void setUserName(String userName) {
this.userName = userName;
} }

3. 创建main入口

package com.test.www;

import java.lang.reflect.Method;

import com.test.www.MethodType.MethodTypeEnum;

public class mainTest {

    public static void main(String[] args) {
Class<Test> testClass = Test.class;
try {
//由于是注解到method上的。所以首先要获取这种方法
Method method = testClass.getDeclaredMethod("getUserName"); //推断这种方法上是否有这个注解
if (method.isAnnotationPresent(MethodType.class)) {
System.out.println("this is a method Annotation"); //假设有这个注解,则获取注解类
MethodType methodType = (MethodType) method.getAnnotation(MethodType.class);
if (methodType != null) {
if (MethodTypeEnum.TYPE1.equals(methodType.methodType())) {
System.out.println("this is TYPE1");
} else {
System.out.println("this is TYPE2");
}
}
} else {
System.out.println("this is not a method Annotation");
} } catch (Exception e) {
} }
}

4. 输出:

this is a method Annotation
this is TYPE2

4. 创建一个自己定义注解 - 作用于域

1. 创建一个自己定义注解

package com.test.www;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* 定义一个作用到域上的自己定义注解
* @author zhuli
* @date 2014-7-5
*/
@Documented//文档
@Retention(RetentionPolicy.RUNTIME)//在执行时能够获取
@Target({ ElementType.FIELD })//作用到类的域上面
public @interface FieldAnnotations { public String value() default ""; //使用的时候 @FieldAnnotations(value="xxx") }

2. 创建一个使用注解的类

package com.test.www;

import com.test.www.MethodType.MethodTypeEnum;

/**
* 一个注解的測试类
* @author zhuli
* @date 2014-7-5
*/
//注入注解作用于类上面
//能够通过反射 获取类的信息之后 获取得到这个注解的值
@UserNameAnnotations(value = "initphp")
public class Test { @FieldAnnotations(value="zhuli")
private String userName; //注解到
@MethodType(methodType=MethodTypeEnum.TYPE2)
public String getUserName() {
return userName;
} public void setUserName(String userName) {
this.userName = userName;
} }

3. 创建main入口类

package com.test.www;

import java.lang.reflect.Field;

public class mainTest {

    public static void main(String[] args) {
Test test = new Test();
Class<Test> testClass = Test.class;
try {
//由于是注解到Field上的。所以首先要获取这个字段
Field field = testClass.getDeclaredField("userName"); //推断这个Field上是否有这个注解
if (field.isAnnotationPresent(FieldAnnotations.class)) {
System.out.println("this is a field Annotation"); //假设有这个注解,则获取注解类
FieldAnnotations fieldAnnotations = (FieldAnnotations) field.getAnnotation(FieldAnnotations.class);
if (fieldAnnotations != null) {
//通过反射给私有变量赋值
field.setAccessible(true);
field.set(test, fieldAnnotations.value());
System.out.println("value:" + test.getUserName());
}
} else {
System.out.println("this is not a field Annotation");
} } catch (Exception e) {
} }
}

4. 输出:

this is a field Annotation
value:zhuli

Java深入 - 深入 Java自己定义注解的更多相关文章

  1. Java注解与自己定义注解处理器

    动机 近期在看ButterKnife源代码的时候.竟然发现有一个类叫做AbstractProcessor,并且ButterKnife的View绑定不是依靠反射来实现的,而是使用了编译时的注解,自己主动 ...

  2. 深入理解Java:注解(Annotation)自己定义注解入门

    深入理解Java:注解(Annotation)自己定义注解入门 要深入学习注解.我们就必须能定义自己的注解,并使用注解,在定义自己的注解之前.我们就必须要了解Java为我们提供的元注解和相关定义注解的 ...

  3. 【面试加分项】java自己定义注解之申明注解

    之前的博客http://blog.csdn.net/u010590685/article/details/47029447介绍了java的注解的基本知识今天我们学习怎样使用自己定义注解. 首先我们要声 ...

  4. 【面试加分项】java自己定义注解之解析注解

    我之前的博客中说明过自己定义注解的声明今天我们来看看怎样对我们自己定义的注解进行使用. 1.我们在程序中使用我们的注解. 上一篇中我们自己定义了一个注解: @Target(ElementType.FI ...

  5. 【java】细说 JAVA中 标注 注解(annotation)

    Java注解是附加在代码中的一些元信息,用于一些工具在编译.运行时进行解析和使用,起到说明.配置的功能.注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用 下面我们来详细说说这个注解,到底是怎么一 ...

  6. java编程思想-java注解

    注解(也被称为元数据)为我们在代码中添加信息提供了一种形式化的方法,使我们可以在稍后某个时刻非常方便的使用这些数据. 一.定义注解 注解的定义看起来很像接口的定义.事实上,与其他任何Java接口一样, ...

  7. Java编程思想学习(十五) 注解

    注解Annotation又叫元数据,是JDK5中引入的一种以通用格式为程序提供配置信息的方式.使用注解Annotation可以使元数据写在程序源码中,使得代码看起来简洁,同时编译器也提供了对注解Ann ...

  8. [Effective Java]第六章 枚举和注解

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  9. [改善Java代码]推荐使用枚举定义常量

    枚举和注解都是在Java1.5中引入的,虽然他们是后起之秀,但是功能不容小觑,枚举改变了常量的声明方式,注解耦合了数据和代码. 建议83:推荐使用枚举定义常量 一.分析 常量的声明是每一个项目中不可或 ...

随机推荐

  1. 【配置】Spring和MyBatis整合

    spring配置文件: <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="h ...

  2. MySQL5.7 GTID在线开启与关闭【转】

    当前场景   当前某些业务还有未开启GTID服务组,升级5.7后,如何检测是否符合开启GTID条件,如何在线修改切换使用GTID:已经升级5.7后,已经开启GTID,如何快速回滚后退: 线上gtid如 ...

  3. selnium远程机上传图片遇到的坑

    一般上传图片方法采取方案如下: input标签的file类型上传图片,使用对象的sendkeys+路径方法 使用js注入,再用使用对象的sendkeys+路径方法 使用autolt生成的exe,打开对 ...

  4. 删除MySQL binlog日志的方法

    对于比较繁忙的OLTP(在线事务处理)系统,由于每天生成日志量大,这些日志如果长时间不清除,将会对磁盘空间带来很大的浪费.因此,定期删除日志是DBA维护MySQL数据库的一个重要工作内容.下面跟大家分 ...

  5. php常用的安全过滤函数

    目录结构 ①常用的安全函数有哪些: ②这些函数的作用: ③函数的用法: ④举例说明: ⑤参考资料: 由于越来越多的项目开始使用框架,所以,很多的程序员也不在关心安全的问题!因为框架已经帮我们几乎完美的 ...

  6. Java编程的逻辑 (61) - 内存映射文件及其应用 - 实现一个简单的消息队列

    本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http:/ ...

  7. ps不显示命令本身的进程号

    当我们查看某个服务的进程时候,它会把命令本身的进程显示出来.如下图: 进程号2383 就是我命令本身的进程号,和我实际想看的进程无关 特别是在我们写脚本,kill进程时候会报错: 解决办法可以优化脚本 ...

  8. 【LOJ】#2548. 「JSOI2018」绝地反击

    题解 卡常卡不动,我自闭了,特判交上去过了 事实上90pts= = 我们考虑二分长度,每个点能覆盖圆的是一段圆弧 然后问能不能匹配出一个正多边形来 考虑抖动多边形,多边形的一个端点一定和圆弧重合 如果 ...

  9. ref:linux用户和组管理,/etc/passwd,/etc/shadow和/etc/group 文件内容解释

    ref:https://www.cnblogs.com/xuha0/p/5519232.html 与用户相关的系统配置文件主要有/etc/passwd 和/etc/shadow,其中/etc/shad ...

  10. IntellijIDEA快速入门(Windows版)

    跟随公司变更技术堆栈的步伐,开始学习相应工具IntelliJ的使用,之前一个大神同时也提到,最近该IDE的市场份额已然超越了免费的Eclipse,因此该工具已经到了必须会的程度了. 新年快乐,鸡年大吉 ...