概念

Java有五个元注解,自动继承java.lang.annotation.Annotation。

什么是元注解,可以理解为其他普通注解进行解释说明

@Target  该注解的使用范围,限定应用场景。枚举类 ElemenetType 中

  1. TYPE:类,接口
  2. FIELD:字段,枚举的常量
  3. METHOD:函数(方法)
  4. PARAMETER:参数
  5. CONSTRUCTOR:构造函数
  6. ANNOTATION_TYPE:注解类型
  7. LOCAL_VARIABLE:局部变量
  8. PACKAGE:包

@Retention  该注解的生存周期,相当于时间戳。枚举类型 RetentionPolicy 中

  1. SOURCE:在源文件中有效,编译后会被丢弃(如@Override,@Deprecated)
  2. CLASS:在class文件中有效,在jvm丢弃
  3. RUNTIME:在运行时有效,class文件保留,jvm运行时保留(很多框架运用反射调用)

@Documented

javadoc文档生成工具的使用

@Inherited

允许子类继承父类中的注解。

@Repeatable

同一种注解可多次使用

作用

  1. 注释,解释,通过代码的标识元数据生成doc文档;
  2. 使用反射,通过代码标识的元数据对代码进行分析;
  3. 编译检查,通过代码标识的元数据让编译器进行基本检查。

实例

定义一个普通的注解,

public @interface Test
{ }

使用我们自定义的注解

@Test
public class Person
{
@Test
private String name; @Test
private int age; @Test
public void say()
{
System.out.println("Hello,Java Annotation");
}
}

@Target(ElementType.PACKAGE) 注解作用的目标>包

这个注解要理解什么是友好声明类和包常量,包中有很多的内部访问的类或常量,就可以统一的放到友好声明类中,这样就方便,而且集中管理,减少friendly类到处游走的情况。

可以参考这个 https://www.cnblogs.com/DreamDrive/p/5428573.html

import java.lang.annotation.ElementType;
@Target(ElementType.PACKAGE)
public @interface Test
{
}
@Test
package test; class Person
{ private String name; private int age; public void say()
{
System.out.println("Hello,Java Annotation");
}
}

@Target(ElementType.CONSTRUCTOR) 注解作用的目标>构造函数

package test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Target;
@Target(ElementType.CONSTRUCTOR)
public @interface Test
{ }
package test;

public class Person
{ private String name; private int age;
@Test
public Person()
{ } public void say()
{
System.out.println("Hello,Java Annotation");
}
}

其他范围就不一一列举,都是相同的。

@Retention(RetentionPolicy.RUNTIME)   生存周期

代码运行时动态获取注解的信息

package test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target(ElementType.CONSTRUCTOR)
@Retention(RetentionPolicy.RUNTIME)
public @interface Test
{ }
package test;

public class Person
{ private String name; private int age;
@Test
public Person()
{ } public void say()
{
System.out.println("Hello,Java Annotation");
}
}

注解的属性-->成员变量

方法名是成员变量的的名字,变量的类型是他的返回值。

package test;

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 Test
{
public int age() default ; public String name() default ""; public String className();
}
package test;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Method; @Test(age = 15,name = "zhangsan",className = "高三(3)班")
public class Person
{ private String name; private int age; public Person()
{ } private void say()
{
System.out.println("Hello,Java Annotation");
}
}

既然给了注解,我们要做的工作必然是要提取注解上面的内容,要拿到这些注解就要用到反射。

package test;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Method; @Test(age = 15,name = "zhangsan",className = "高三(3)班")
public class Person
{private String name; private int age; public Person()
{ } private void say()
{
System.out.println("Hello,Java Annotation");
} public static void main(String[] args)
{
boolean hasAnnotation = Person.class.isAnnotationPresent(Test.class); if (hasAnnotation)
{
Test test = Person.class.getAnnotation(Test.class); System.out.println("age:" + test.age());
System.out.println("name:" + test.name());
System.out.println("className:" + test.className());
}
}
}

输出结果

age:15
name:zhangsan
className:高三(3)班

对于类的属性和方法,都是同样的道理。

成员注解

package test;

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 Field
{
public String name();
}

函数注解

package test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Method
{
public String say();
}

Person类

package test;

import java.lang.annotation.Annotation;

@Test(age = 15,name = "zhangsan",className = "高三(3)班")
public class Person
{
@Field(name = "lisi")
private String name; private int age; public Person()
{ }
@Method(say = "hello")
private void say()
{
System.out.println("Hello,Java Annotation");
} public static void main(String[] args)
{
boolean hasAnnotation = Person.class.isAnnotationPresent(Test.class); if (hasAnnotation)
{
Test test = Person.class.getAnnotation(Test.class); System.out.println("age:" + test.age());
System.out.println("name:" + test.name());
System.out.println("className:" + test.className());
} try
{
java.lang.reflect.Field field = Person.class.getDeclaredField("name"); field.setAccessible(true); Field check = field.getAnnotation(Field.class); if (check != null)
{
System.out.println("check value:" + check.name());
} java.lang.reflect.Method method = Person.class.getDeclaredMethod("say"); if (method != null)
{
Annotation[] ans = method.getAnnotations();
for (int i = 0; i < ans.length; i++)
{
System.out.println("method annotation:" + ans[i].annotationType().getSimpleName());
}
} } catch (NoSuchFieldException e)
{
e.printStackTrace();
}catch (Exception e){
e.printStackTrace();
} }
}

输出

age:15
name:zhangsan
className:高三(3)班
check value:lisi
method annotation:Method

可以看出我们获取了注解上的值,现在都没有实际意义,我们可以用注解来做些什么?

这些我没办法给出确切答案,只能说根据各人需求去合理利用注解。

实例

注解参数赋值

@Test注解

package test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface Test
{
String value();
}

Person类属性赋值

package test;

import org.apache.poi.ss.formula.functions.T;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Method; public class Person
{
@Test("zhangsan")
private String name; @Test("15")
private int age; public Person()
{ } private void say()
{
System.out.println("Hello,Java Annotation");
} @Override
public String toString()
{
return "name = " + name + "\n" + "age = " + age;
} public static void main(String[] args)
{
Person person = new Person();
try
{
//取得成员变量的值
Field field = Person.class.getDeclaredField("name");
//打开权限
field.setAccessible(true);
//判断属性是否有注解
if (field.isAnnotationPresent(Test.class))
{
//获取属性上的注解值
Test test = field.getAnnotation(Test.class);
String name = test.value();
//赋值
field.set(person,name);
} Field field1 = Person.class.getDeclaredField("age");
field1.setAccessible(true);
if (field1.isAnnotationPresent(Test.class))
{
Test test = field1.getAnnotation(Test.class);
int age = Integer.valueOf(test.value());
field1.set(person,age);
}
System.out.println(person); }catch (Exception e)
{
e.printStackTrace();
} }
}

输出

name = zhangsan
age = 15

用注解去检查函数等等。。

lz不善于用语言表达,所以可能有很多人没看懂,可以看下

一个大佬讲解的注解说明(简单易懂)

https://blog.csdn.net/briblue/article/details/73824058

Java的自定义注解使用实例的更多相关文章

  1. Java实现自定义注解开发

    Java实现自定义注解开发 一直都对注解开发挺好奇的,最近终于有时间自己实践了一把,记录一下 万一后期会用到呢 哈哈哈 首先我们了解一下自定义注解的标准示例,注解类使用 @interface 关键字修 ...

  2. java自定义注解知识实例及SSH框架下,拦截器中无法获得java注解属性值的问题

    一.java自定义注解相关知识 注解这东西是java语言本身就带有的功能特点,于struts,hibernate,spring这三个框架无关.使用得当特别方便.基于注解的xml文件配置方式也受到人们的 ...

  3. 注解:java 自定义注解应用实例

    本例子旨在使用自定义注解为实体打上标记,为自动生成 sql 提供依据,模拟 hibernate 的注解,至于注解的原理自己搜吧 1.定义 Table 注解 package test; import j ...

  4. [转] Java @interface 自定义注解

    [From] http://blog.csdn.net/afterlife_qiye/article/details/53748973 1. 注解的好处 注解可以替代配置文件完成对某些功能的描述,减少 ...

  5. Java Annotation自定义注解详解

    在开发过程中总能用到注解,但是从来没有自己定义过注解.最近赋闲在家,研究整理了一番,力求知其然知其所以然. 本文会尝试描述什么是注解,以及通过一个Demo来说明如何在程序中自定义注解.Demo没有实际 ...

  6. Java利用自定义注解、反射实现简单BaseDao

    在常见的ORM框架中,大都提供了使用注解方式来实现entity与数据库的映射,这里简单地使用自定义注解与反射来生成可执行的sql语句. 这是整体的目录结构,本来是为复习注解建立的项目^.^ 好的,首先 ...

  7. Java基于自定义注解的面向切面的实现

    目的:实现在任何想要切的地方添加一个注解就能实现面向切面编程 自定义注解类 @Target({ElementType.PARAMETER, ElementType.METHOD}) @Retentio ...

  8. java学习--自定义类的实例的大小比较和排序

    我们知道Object类有一个equals方法,用于比较两个对象是否相等 我们只要在自定义类中重写了equals方法(若不重写就是比较两个实例的地址,相当于==)就可以用来比较该类的两个实例是否相等 问 ...

  9. java/springboot自定义注解实现AOP

    java注解 即是注释了,百度解释:也叫元数据.一种代码级别的说明. 个人理解:就是内容可以被代码理解的注释,一般是一个类. 元数据 也叫元注解,是放在被定义的一个注解类的前面 ,是对注解一种限制. ...

随机推荐

  1. SharePoint2013 列表栏设置

    在实际项目中,会遇到对列表栏的深度操作,比如设置在新建项目也就是newForm是否可见,是否有默认值,默认标题等等,这类深度操作在页面上是无法配置的,因为需要设置SPFild这个对象,但是用share ...

  2. JavaScript程序的执行顺序

    JavaScript程序的执行顺序:同步==>异步==>回调 同步是阻塞模式,异步是非阻塞模式.     同步就是指一个进程在执行某个请求的时候,若该请求需要一段时间才能返回信息,那么这个 ...

  3. 物流的纯css实现方法

    首先我们来看看UI给出的设计图. 为什么到达是最前面,为什么物流顺序是倒叙的,这是什么用户习惯,这是我拿到设计稿的问题,但是这里不谈设计,因为审美这个东西无法评估.那么这里我就做一个顺序的来对比一下吧 ...

  4. Html5的表单元素

    表单是HTML中获取用户输入的手段,,对于web应用系统及其重要,文字是不能说明问题的: 直接上代码把: <!DOCTYPE html><html lang="en&quo ...

  5. mysql 存储引擎简介

    几个常用存储引擎的特点 下面我们重点介绍几种常用的存储引擎并对比各个存储引擎之间的区别和推荐使用方式. 特点 Myisam BDB Memory InnoDB Archive 存储限制 没有 没有 有 ...

  6. pycharm linux版快捷方式创建

    ****************************pycharm_linux安装and快捷方式创建******************1.下载好安装包之后解压:    tar -xfz 压缩包名 ...

  7. flock SUSE/RHEL

    Util-linux-2.26 Util-linux 软件包其它实用程序.包括处理文件系统.控制台.分区以及消息等工具. 大概编译时间:1.3 SBU 需要磁盘空间:137 MB 6.65.1. FH ...

  8. String和StringBuffer的区别?

    这个我经常用的是String,说真的,用StringBuffer的次数还真是少,唯一让我觉得特别的方法就是appand这个方法是StringBuffer独有的,那么他们到底有什么区别呢,我们知道Str ...

  9. CLR、程序集、反射和控制反转

    以前面试包括自己学习的时候经常会碰到这3个东西,也查过相关介绍,晦涩难懂,虽然看完之后,当时勉强理解,不过过一段时间又忘了.其实这篇文章可以分两篇(clr.程序集)和(反射.控制反转)来写,但它们之间 ...

  10. XML解析的四种方法 建议使用demo4j解析 测试可以用

    https://www.cnblogs.com/longqingyang/p/5577937.html 4.DOM4J解析  特征: 1.JDOM的一种智能分支,它合并了许多超出基本XML文档表示的功 ...