先引用一下百度百科的名词解析:

定义:注解(Annotation),也叫元数据。一种代码级别的说明。它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面,用来对这些元素进行说明,注释。
作用分类:
①编写文档:通过代码里标识的元数据生成文档【生成文档doc文档】
② 代码分析:通过代码里标识的元数据对代码进行分析【使用反射】
③编译检查:通过代码里标识的元数据让编译器能够实现基本的编译检查【Override】

下面开始正文:

概述:

就如上面所说,注解有几种用途,第一种是编写文档,这种注解会在javadoc中存在;

第二种是代码分析,这需要使用 反射 的api去解析注解;

第三种类是提供编译时的检查,如 deprecated 这些,编译的时候会产生警告信息。

举几个例子:

1、下面的例子,注解使用了@Documented注解,这意味着,我们的注解会在 javadoc 生成的文档中出现,另外一个@Retention的用途下面会说到

package com.ruby;

import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy; @Documented
@Retention(RetentionPolicy.SOURCE)
public @interface ClassPreamble {
String author();
}

  使用:

@ClassPreamble(
author = "ruby"
)
public class Main ...

  

2、代码分析(我们拙劣地模拟一下Spring Boot的路由注解)

代码目录结构:

RequestMapping注解定义:

package com.ruby.annotation;

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.METHOD)
public @interface RequestMapping {
String value() default "/";
}

  

Home控制器定义:

package com.ruby.controller;

import com.ruby.annotation.RequestMapping;

public class Home {
@RequestMapping("test")
public void HelloWorld() {
System.out.println("HelloWorld !");
}
}

使用方法(命名不是太规范,test一般用于测试文件命名):关键是main方法里面通过反射去解析注解的代码,其他两个方法只是工具方法(用以获取包下面所有类)

package com.ruby.test;

import com.ruby.annotation.RequestMapping;

import java.io.File;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List; public class TestHome {
public static void main(String[] args) throws ClassNotFoundException, IOException, InstantiationException, IllegalAccessException, InvocationTargetException{
String route = "test"; Class[] classes = getClasses("com.ruby.controller");
for (Class cls : classes) {
Method[] methods = cls.getMethods();
for (Method method : methods) {
RequestMapping requestMapping = method.getAnnotation(RequestMapping.class);
if (requestMapping != null) {
String mapping = requestMapping.value();
if (mapping.equals(route)) {
method.invoke(cls.newInstance());
}
}
}
}
} /**
* Scans all classes accessible from the context class loader which belong to the given package and subpackages.
*
* @param packageName The base package
* @return The classes
* @throws ClassNotFoundException
* @throws IOException
*/
private static Class[] getClasses(String packageName)
throws ClassNotFoundException, IOException {
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
assert classLoader != null;
String path = packageName.replace('.', '/');
Enumeration<URL> resources = classLoader.getResources(path);
List<File> dirs = new ArrayList<File>();
while (resources.hasMoreElements()) {
URL resource = resources.nextElement();
dirs.add(new File(resource.getFile()));
}
ArrayList<Class> classes = new ArrayList<Class>();
for (File directory : dirs) {
classes.addAll(findClasses(directory, packageName));
}
return classes.toArray(new Class[classes.size()]);
} /**
* Recursive method used to find all classes in a given directory and subdirs.
*
* @param directory The base directory
* @param packageName The package name for classes found inside the base directory
* @return The classes
* @throws ClassNotFoundException
*/
private static List<Class> findClasses(File directory, String packageName) throws ClassNotFoundException {
List<Class> classes = new ArrayList<Class>();
if (!directory.exists()) {
return classes;
}
File[] files = directory.listFiles();
for (File file : files) {
if (file.isDirectory()) {
assert !file.getName().contains(".");
classes.addAll(findClasses(file, packageName + "." + file.getName()));
} else if (file.getName().endsWith(".class")) {
classes.add(Class.forName(packageName + '.' + file.getName().substring(0, file.getName().length() - 6)));
}
}
return classes;
}
}

  

3、编译时检查:如@SuppressWarnings,如有些废弃的方法调用之后会有警告信息,你可以使用@SuppressWarnings注解:@SuppressWarnings("deprecation"),这样编译的时候就不会产生警告信息了。

注解定义

注解定义的方法类似接口定义(上面有例子了),只是在interface关键字前面多了个@符号,注解定义上面也还可以有其他注解(下面详说),注解体里面的元素基本格式都一致,类型名称 + 变量名 + 括号 + default xxx; 默认值的提供是可选的。类型可以是普通类型,也可以是类啊、异常啊这些。

接下来要说的发图吧,比较清晰:

注解分类:

1、普通注解,如:@Deprecated、@Override、@SuppressWarnings、@SafeVarargs等

2、注解其他注解的注解(比较拗口):记重点了,比较关键的是@Retention、@Target

别的不说,先发图,看图比较清晰:

先说@Retention,这个很重要,如果我们要在运行时使用定义的注解,一定不要写错了,默认是不会到运行时还存在的

使用方法:在注解定义处使用:

@Retention(RetentionPolicy.SOURCE)
public @interface ClassPreamble {
String author();
}

有三个可选的选项:(默认是RetentionPolicy.CLASS)

1、RetentionPolicy.SOURCE,如果使用该选项,我们的注解只会在源代码存在,编译之后就没有了,我们去看编译之后的 class 文件,会发现注解已经不存在了。但是如果我们不写@Retention 的话,编译后的 class 文件还会有该注解,因为默认是RetentionPolicy.CLASS。

2、RetentionPolicy.CLASS(默认),使用该选项,我们的注解会在源代码和class文件中存在

3、RetentionPolicy.RUNTIME,使用该选项,我们的注解会一直存在,我们可以在运行的时候去获取注解信息,然后做相应处理。

@Document 这个注解会在 javadoc 生成的文档中出现

@Target,这也算是一个比较重要的注解,指明了该注解可以用在什么地方,好比如,你指定了一个注解的Target为ElementType.METHOD,那么这个注解只能用在方法上,而不能用在field或者type上,当然你也可以同时指定多个,如:

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

 可用类型如上图。

@Repeatable 指明该注解是否可以多次使用

好了,就这么多了,其他还不是太了解,上面只是看文档总结的东西,

其实最重要的还是@Retention,如果要在运行时也用到注解信息,那么必须指定  Retention.RUNTIME 参数,否则通过反射去获取注解是获取不到的。

与此相关的另外的知识就是反射了,有空再补充。

java 注解详解的更多相关文章

  1. 《Java基础知识》Java注解"@"详解

    Java注解含义: Java注解,顾名思义,注解,就是对某一事物进行添加注释说明,会存放一些信息,这些信息可能对以后某个时段来说是很有用处的.Java注解又叫java标注,java提供了一套机制,使得 ...

  2. 【Java基础】Java 注解详解

    对于Java注解,我之前的印象是很模糊的,总觉得这个东西经常听说,也经常用,但是具体是怎么回事,好像没有仔细学习过,说到注解,立马想到@Controller,仅此而已. 对于Java注解,我咨询过一些 ...

  3. Java注解详解

    Java1.5引入了注解,注解作为程序的元数据嵌入到程序中.注解可以被一些解析工具或者编译工具进行解析.我们也可以声明注解在编译过程或者执行时产生作用. 创建Java自定义注解: package co ...

  4. Java基础13:反射与注解详解

    Java基础13:反射与注解详解 什么是反射? 反射(Reflection)是Java 程序开发语言的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性. Orac ...

  5. java中的注解详解和自定义注解

    一.java中的注解详解 1.什么是注解 用一个词就可以描述注解,那就是元数据,即一种描述数据的数据.所以,可以说注解就是源代码的元数据.比如,下面这段代码: @Override public Str ...

  6. Java Servlet详解(体系结构+注解配置+生命周期)

    Java Servlet详解(注解配置+生命周期) 什么是Servlet : (Server applet)? 顾名思义:服务端的小程序 Servlet只是一个接口,定义了Java被浏览器访问到(To ...

  7. Java Annotation详解 理解和使用Annotation

    系统中用到了java注解: 查了一下如何使用注解,到底注解是什么: (1)创建方法:MsgTrace Java Class==> 在Create New Class中: name:输入MsgTr ...

  8. @RequestBody, @ResponseBody 注解详解(转)

    原文地址: https://www.cnblogs.com/qq78292959/p/3760651.html @RequestBody, @ResponseBody 注解详解(转) 引言: 接上一篇 ...

  9. SpringMVC 常用注解 详解

    SpringMVC 常用注解 详解 SpringMVC 常用注解 1.@RequestMapping                                      路径映射 2.@Requ ...

随机推荐

  1. 4星|《财经》2018年第13期:年轻人大多从大三和大四起开始就从QQ向微信转移

    <财经>2018年第13期 总第530期 旬刊 本期主要话题是快递业,其他我感兴趣的重要话题还有:香港9价HPV疫苗断供风波:华盛顿邮报被贝佐斯收购后这几年的变化:北京二中朝阳学校的划片风 ...

  2. leetcode28_C++实现strStr()函数

    实现 strStr() 函数. 给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始).如果不存在,则返 ...

  3. Controller控制器

    控制器概述 上接应用(北向),下接设备(南向),左右扩展(东西接口). 谁掌控了控制器,或者制定了标准,就掌握了SDN. 南向功能:通过Openflow等南向接口技术,对网络设备进行管控,例如拓扑发现 ...

  4. struts2--文件上传大小

    Struts2文件上传的大小限制问题 问题:上传大文件报错-- 解决:修改struts.xml文件中的参数如下 <constant name="struts.multipart.max ...

  5. android入门 — 多线程(一)

    android中的一些耗时操作,例如网络请求,如果不能及时响应,就会导致主线程被阻塞,出现ANR,非常影响用户体验,所以一些耗时的操作,我们会想办法放在子线程中去完成. android的UI操作并不是 ...

  6. [2017BUAA软工]第三次博客作业:案例分析

    第三次博客作业:案例分析 1. 调研和评测 1.1 BUG及设计缺陷描述 主要测试博客园在手机端上的使用情况. [BUG 01] 不能后退到上一界面(IOS) 重现步骤:打开博客首页中任意博文,点击博 ...

  7. 软件工程个人作业3——集大通APP案例分析

    第一部分:调研, 评测 1.第一次上手体验 主要界面截图: 感受: 1.界面不美观: 2.特色功能展现模块不突出,以上截图为打开APP所看到的界面展示,但是这些功能都不是该APP的特色功能,显得有些累 ...

  8. LR之Java Vuser II

    最近项目待压测的服务端协议使用的是java的Netty框架开发,而传输的业务数据使用了google protobuf进行序列化,然后通过tcp数据流与客户端通讯.这一次的压测脚本决定使用LR的java ...

  9. virtualenv是什么?virtualenv的安装及pycharm的配置和使用

    virtualenv是什么? virtualenv是一个创建隔绝的Python环境的工具.virtualenv创建一个包含所有必要的可执行文件的文件夹,用来使用Python工程所需的包.简单的说就是一 ...

  10. 解决 Package test is missing dependencies for the following libraries: libcrypto.so.1.0.0

    根据项目要求需要用到openssl这个库,看了看编译环境幸好本身就集成了该库.但在编译openssl的功能时,碰到缺少类库的错误. Package test is missing dependenci ...