java之美妙的注解
注解
一、认识注解
先看百度百科对java注解的解释:
定义:注解(Annotation),也叫元数据。一种代码级别的说明。它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面,用来对这些元素进行说明,注释。
作用分类:
①编写文档:通过代码里标识的元数据生成文档【生成文档doc文档】
② 代码分析:通过代码里标识的元数据对代码进行分析【使用反射】
③编译检查:通过代码里标识的元数据让编译器能够实现基本的编译检查【Override】
提到的元数据,可以理解为描述数据的数据,看起来很绕,元注解也是描述注解的注解,后面说。
Annotation都是java.lang.annotation.Annotation接口的子接口,注解是一种特殊的接口,从代码格式上来看确实是,是在interface前加了一个@
符号,格式为@interface xxx{}
二、JDK内置注解
jdk就存在注解,比如我们用eclipse子类覆盖父类方法时候用快捷间alt+/会自带在方法名称上面贴上一个@Override的标签。
@Override
这个就是在子类覆盖父类的方法时候,经常遇到
@Deprecated
这个元素是用来标记过时的元素,想必大家在日常开发中经常碰到。编译器在编译阶段遇到这个注解时会发出提醒警告,告诉开发者正在调用一个过时的元素比如过时的方法、过时的类、过时的成员变量。 如java.util.Data类中有很多过时的方法
@SupressWarings
抑制警告,有很多人有代码洁癖,看到黄色的警告不爽,用它就可以抑制住。
@SafeVarargs
java7出现的,抑制堆污染警告,有点自欺欺人的感觉。
三、元注解
元注解就是注解的注解,来看一下有哪些元注解
@Retention
Retention英文是保留的意思,在这里可以约束注解的存活周期,代码运行有三个周期,分别为Source(源代码)、Class(字节码)、Runtime(运行时期) 三个时期的值保留在RetentionPolicy这个枚举类中。所以我们可以这样来玩,
@Retention(RetetionPolicy.常量值)//RESOURCE,CLASS,RUNTIME
@Target
Target是目标的意思,这里约束这个注解在那可以贴(类,方法,构造器,参数等),位置的常量在ElementType这个枚举类中
@Target({ElementsType.常亮值})//TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE, TYPE表示类,接口,枚举
@Documented
Document文档的意思,也就是使用这个标签到时候会带到API文档中去
@Inherited
Inherited是遗传的意思,顾名思义,这个注解会遗传到子类去
@Repeatable 可重复的,Java8的一个新特性
四、自定义注解
1.注解的属性
在注解中定义属性,必须是8 种基本数据类型外加 类、接口、注解及它们的数组。不能是Integer..注解中属性可以有默认值,默认值需要用 default 关键值指定。
如:
@interface TestAnnotation{
int id() default 0;
String msg() default "hello";
}
@TestAnnotation(id = 3,msg = "aa")
class Test1{
}
//当注解中只有一个value属性时候,贴标签的时候可以省略value。
//当注解中没有属性时,括号都可以省略(override注解)
@interface Test1Annotation{
int[] value();
}
@Test1Annotation({1,2})
class Test2{
}
2.用反射操作注解
注解可以在类Class,方法Method,字段Field,构造器Constructor上等,所以在各自的类中都存在获取注解的API
boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)
如果指定类型的注释存在于此元素上,则返回 true,否则返回 false。
Annotation[] getAnnotations() 返回此元素上存在的所有注释。
<Annotation> getAnnotation(Class annotationClass)如果存在该元素的指定类型的注释,则返回这些注释对象,否则返回 null。 获取注解对象
自定义一个注解:
@Target({TYPE, FIELD, METHOD})//注解可以放的目标
@Retention(RetentionPolicy.RUNTIME)//注解保留周期
public @interface AnnotationTest {
int id();
String name();
}
测试一下这个自定义的注解:
@AnnotationTest(id = 1,name = "na")
class Test{
public static void main(String[] args) throws Exception {
//判断Test(类元素)是否应用了这个注解
boolean hasAnnotation = Test.class.isAnnotationPresent(AnnotationTest.class);
if(hasAnnotation){
//获取注解对象
AnnotationTest t = Test.class.getAnnotation(AnnotationTest.class);
System.out.println(t.name());
System.out.println(t.id());
//获取所有的注解s
Annotation[] ans = Test.class.getAnnotations();
for (Annotation an : ans) {
System.out.println(an);
}
}
System.out.println("============");
//获取方法中上的注解
boolean has = Test.class.getMethod("work").isAnnotationPresent(AnnotationTest.class);
if(has){
AnnotationTest an = Test.class.getMethod("work").getAnnotation(AnnotationTest.class);
System.out.println(an.name());
}
}
@AnnotationTest(id = 0,name="hello")
public void work(){
}
}
结果为:
na
1
@annotation.AnnotationTest(id=1, name=na)
============
hello
五、注解的用途
实践:模拟Junit4.x
Junit是做单元测试的,junit4用了注解,用法是在一个公共无参的方法贴上一个@Test标签表示要测试这个方法,但是在测试之前有可能会出现一些初始化操作,在结束后可能需要释放一些资源,需要初始化的方法上面贴上一个@Before,Test之后需要执行的需要贴上@After。
现在我们来自定义注解来模拟一下这些操作,需要在这里提的是,其实注解本身没什么意义,需要人为的设置它完成一些功能。
我们先定义三个注解:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAfter {
}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyBefore {
}
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyTest {
}
来一个测试这些注解功能的类
public class Employee {
@MyBefore
public void init(){
System.out.println("-------初始化-------");
}
@MyAfter
public void end(){
System.out.println("-------销毁---------");
}
@MyTest
public void save() {
System.out.println("保存操作");
}
@MyTest
public void delete() {
System.out.println("删除操作");
}
@MyBefore
public void ddd(){
System.out.println("怎么说呢?");
}
}
好了,接下来我们就得具体去设置这个注解的意义
思路和步骤:
获取Employee类的字节码对象,得到所有的方法
- 对所有方法进行迭代,将三类注解分别存储,把有@MyBefore注解的存在beforeList,带有@MyAfter存在afterList,带有@MyTest存在testList
对testList集合迭代,在迭代 过程中先执行beforeList中的方法,在执行afterList中的方法
- 对所有方法进行迭代,将三类注解分别存储,把有@MyBefore注解的存在beforeList,带有@MyAfter存在afterList,带有@MyTest存在testList
public class JunitTest {
public static void main(String[] args) throws Exception {
//得到Employee中所有的方法
Method[] mds = Employee.class.getDeclaredMethods();
//创建Employee的对象,用来执行方法
Employee e = Employee.class.newInstance();
//用集合存储三类注解标注的方法
List<Method> beforeList = new ArrayList<>();
List<Method> afterList = new ArrayList<>();
List<Method> testList = new ArrayList<>();
//对所有的方法进行迭代,分类
for (Method md : mds) {
if(md.isAnnotationPresent(MyBefore.class)){
beforeList.add(md);
}else if(md.isAnnotationPresent(MyAfter.class)){
afterList.add(md);
}else if(md.isAnnotationPresent(MyTest.class)){
testList.add(md);
}
}
//在对test迭代的过程中,先执行所有被before注解了的方法,然后执行test注解的方法,最后再执行after注解了的方法
for (Method method : testList) {
for (Method before : beforeList) {
before.invoke(e); //方法都没有参数
}
method.invoke(e);
for (Method after : afterList) {
after.invoke(e);
}
}
}
}
结果:
-------初始化-------
怎么说呢?
删除操作
-------销毁---------
-------初始化-------
怎么说呢?
保存操作
-------销毁---------
java之美妙的注解的更多相关文章
- Java基础笔记 – Annotation注解的介绍和使用 自定义注解
Java基础笔记 – Annotation注解的介绍和使用 自定义注解 本文由arthinking发表于5年前 | Java基础 | 评论数 7 | 被围观 25,969 views+ 1.Anno ...
- Java学习记录-注解
注解 一.org.springframework.web.bind.annotation ControllerAdviceCookieValue : 可以把Request header中关于cooki ...
- JUnit 4 使用 Java 5 中的注解(annotation)
JUnit 4 使用 Java 5 中的注解(annotation),以下是JUnit 4 常用的几个 annotation 介绍@Before:初始化方法@After:释放资源@Test:测试方法, ...
- Java中万恶的注解
本文由码农网 – 孙腾浩原创翻译,转载请看清文末的转载要求,欢迎参与我们的付费投稿计划! 当Java 1.5引入注解,企业开发者对简化EJB和其他企业产品开发抱有很大期望.可以看一看同一时期的一篇文章 ...
- Java 8 可重复注解与类型注解
Java 8 可重复注解与类型注解 Java 8 对注解处理提供了两点改进:可重复的注解及可用于类型的注解. // 首先要提供一个容器,MyAnnotation 才能用于可重复注解 @Target({ ...
- 转: JaxbContext生成xml文件或java类对象转化注解
JAXB(Java API for XML Binding),提供了一个快速便捷的方式将Java对象与XML进行转换.在JAX-WS(Java的WebService规范之一)中,JDK1.6 自带的版 ...
- Java基础教程(16)--注解
一.注解基础知识 1.注解的格式 最简单的注解就像下面这样: @Entity @符号指示编译器其后面的内容是注解.在下面的例子中,注解的名称为Override: @Override void ...
- Java基础教程:注解
Java基础教程:注解 本篇文章参考的相关资料链接: 维基百科:https://zh.wikipedia.org/wiki/Java%E6%B3%A8%E8%A7%A3 注解基础与高级应用:http: ...
- Java复习——枚举与注解
枚举 枚举就是让某些变量的取值只能是若干固定值中的一个,否则编译器就会报错,枚举可以让编译器在编译阶段就控制程序的值,这一点是普通变量无法实现的.枚举是作为一种特殊的类存在的,使用的是enum关键字修 ...
随机推荐
- zw版【转发·台湾nvp系列Delphi例程】HALCON SubImage
zw版[转发·台湾nvp系列Delphi例程]HALCON SubImage procedure TForm1.Button1Click(Sender: TObject);var op : HOper ...
- Java 动态代理是基于什么原理
动态代理 ①动态代理概念理解 动态代理是一种方便运行时动态构建代理.动态处理代理方法调用的机制,很多场景都利用类似机制做到的,比如用来包装RPC调用.面向切面的变成(AOP) 实现动态代理的方式很多, ...
- C#——文件上传(一般处理程序ashx)
Framework版本:.Net Framework 4 1.FileInfo实体 using System; using System.Collections.Generic; using Syst ...
- deepin中crossover或playonlinux装完office后word无法输入中文的问题
原因:office安装是自带了一个微软输入法 解决:装offce时进行自定义安装,在office共享功能里,把输入法去掉. 参考: https://jingyan.baidu.com/article/ ...
- map.containsKey
该方法判断Map集合对象中是否包含指定的键名.如果Map集合中包含指定的键名,则返回true,否则返回false. 语法 containsKey(Object key) . e.g public s ...
- 「翻译」一篇redis文章引发的翻译——JVM能支持多少线程?
昨天看了一篇关于redis 的文章https://www.cnblogs.com/fanwencong/p/5782860.html 作者说他模拟了100万线程的并发,我对这个有一些怀疑,看了评论也有 ...
- SPOJ—VLATTICE Visible Lattice Points(莫比乌斯反演)
http://www.spoj.com/problems/VLATTICE/en/ 题意: 给一个长度为N的正方形,从(0,0,0)能看到多少个点. 思路:这道题其实和能量采集是差不多的,只不过从二维 ...
- __autoload自动加载类
在使用PHP的OO模式开发系统时,通常大家习惯上将每个类的实现都存放在一个单独的文件里,这样会很容易实现对类进行复用,同时将来维护时也很便利.这也是OO设计的基本思想之一.在PHP5之前,如果需要使用 ...
- 何时使用MQ ?
何时使用MQmq作为一种基础中间件在互联网项目中有着大量的使用. 一种技术的产生自然是为了解决某种需求,通常来说是以下场景: 需要跨进程通信:B系统需要A系统的输出作为输入参数.当A系统的输出能力远远 ...
- 为什么要使用MQ消息中间件?它解决了什么问题?
1.应用场景 1.1 异步处理 场景说明:用户注册后,需要发注册邮件和注册短信,传统的做法有两种1.串行的方式;2.并行的方式 (1)串行方式:将注册信息写入数据库后,发送注册邮件,再发送注册短信,以 ...