你分析过@Annotation注解的实现原理吗?
点击上方“方志朋”,选择“设为星标”
回复”666“获取新整理的面试文章
作者:阿丙
主页:www.cnblogs.com/acm-bingzi/
什么是注解?
对于很多初次接触的开发者来说应该都有这个疑问?Annontation
是Java5开始引入的新特征,中文名称叫注解。它提供了一种安全的类似注释的机制,用来将任何的信息或元数据(metadata)与程序元素(类、方法、成员变量等)进行关联。为程序的元素(类、方法、成员变量)加上更直观更明了的说明,这些说明信息是与程序的业务逻辑无关,并且供指定的工具或框架使用。
Annontation
像一种修饰符一样,应用于包、类型、构造方法、方法、成员变量、参数及本地变量的声明语句中。
Java注解是附加在代码中的一些元信息,用于一些工具在编译、运行时进行解析和使用,起到说明、配置的功能。注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用。包含在 java.lang.annotation
包中。
注解的用处:
1、生成文档。这是最常见的,也是java 最早提供的注解。常用的有@param
@return
等
2、跟踪代码依赖性,实现替代配置文件功能。比如Dagger 2
依赖注入,未来java 开发,将大量注解配置,具有很大用处;
3、在编译时进行格式检查。如@override
放在方法前,如果你这个方法并不是覆盖了超类方法,则编译时就能检查出。
注解的原理:
注解本质是一个继承了Annotation
的特殊接口,其具体实现类是Java 运行时生成的动态代理类。而我们通过反射获取注解时,返回的是Java 运行时生成的动态代理对象$Proxy1
。通过代理对象调用自定义注解(接口)的方法,会最终调用AnnotationInvocationHandler
的invoke
方法。该方法会从memberValues
这个Map 中索引出对应的值。而memberValues
的来源是Java 常量池。
元注解:
java.lang.annotation
提供了四种元注解,专门注解其他的注解(在自定义注解的时候,需要使用到元注解):
@Documented
– 注解是否将包含在JavaDoc中@Retention
– 什么时候使用该注解@Target
– 注解用于什么地方@Inherited
– 是否允许子类继承该注解
1.@Retention – 定义该注解的生命周期
RetentionPolicy.SOURCE
: 在编译阶段丢弃。这些注解在编译结束之后就不再有任何意义,所以它们不会写入字节码。@Override
,@SuppressWarnings
都属于这类注解。RetentionPolicy.CLASS
: 在类加载的时候丢弃。在字节码文件的处理中有用。注解默认使用这种方式RetentionPolicy.RUNTIME
: 始终不会丢弃,运行期也保留该注解,因此可以使用反射机制读取该注解的信息。我们自定义的注解通常使用这种方式。
2.Target – 表示该注解用于什么地方。默认值为任何元素,表示该注解用于什么地方。可用的ElementType 参数包括
ElementType.CONSTRUCTOR
: 用于描述构造器ElementType.FIELD
: 成员变量、对象、属性(包括enum实例)ElementType.LOCAL_VARIABLE
: 用于描述局部变量ElementType.METHOD
: 用于描述方法ElementType.PACKAGE
: 用于描述包ElementType.PARAMETER
: 用于描述参数ElementType.TYPE
: 用于描述类、接口(包括注解类型) 或enum声明
3.@Documented – 一个简单的Annotations 标记注解,表示是否将注解信息添加在java 文档中。
4.@Inherited – 定义该注释和子类的关系
@Inherited
元注解是一个标记注解,@Inherited
阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited
修饰的annotation
类型被用于一个class,则这个annotation
将被用于该class 的子类。
常见标准的Annotation:
1.Override
java.lang.Override
是一个标记类型注解,它被用作标注方法。它说明了被标注的方法重载了父类的方法,起到了断言的作用。如果我们使用了这种注解在一个没有覆盖父类方法的方法时,java 编译器将以一个编译错误来警示。
2.Deprecated
Deprecated
也是一种标记类型注解。当一个类型或者类型成员使用@Deprecated
修饰的话,编译器将不鼓励使用这个被标注的程序元素。所以使用这种修饰具有一定的“延续性”:如果我们在代码中通过继承或者覆盖的方式使用了这个过时的类型或者成员,虽然继承或者覆盖后的类型或者成员并不是被声明为@Deprecated
,但编译器仍然要报警。
3.SuppressWarnings
SuppressWarning
不是一个标记类型注解。它有一个类型为String[]
的成员,这个成员的值为被禁止的警告名。对于javac 编译器来讲,被-Xlint
选项有效的警告名也同样对@SuppressWarings
有效,同时编译器忽略掉无法识别的警告名。
@SuppressWarnings("unchecked")
自定义注解:
自定义注解类编写的一些规则:
Annotation
型定义为@interface
, 所有的Annotation
会自动继承java.lang.Annotation
这一接口,并且不能再去继承别的类或是接口.参数成员只能用public 或默认(default) 这两个访问权修饰
参数成员只能用基本类型byte、short、char、int、long、float、double、boolean八种基本数据类型和String、Enum、Class、annotations等数据类型,以及这一些类型的数组.
要获取类方法和字段的注解信息,必须通过Java的反射技术来获取 Annotation 对象,因为你除此之外没有别的获取注解对象的方法
注解也可以没有定义成员,不过这样注解就没啥用了
PS:自定义注解需要使用到元注解
自定义注解实例:
FruitName.java
import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;/*** 水果名称注解*/
@Target(FIELD)
@Retention(RUNTIME)
@Documented
public @interface FruitName {String value() default "";
}
FruitColor.java
import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;/*** 水果颜色注解*/
@Target(FIELD)
@Retention(RUNTIME)
@Documented
public @interface FruitColor {/*** 颜色枚举*/public enum Color{ BLUE,RED,GREEN};/*** 颜色属性*/Color fruitColor() default Color.GREEN;}
FruitProvider.java
import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;/*** 水果供应者注解*/
@Target(FIELD)
@Retention(RUNTIME)
@Documented
public @interface FruitProvider {/*** 供应商编号*/public int id() default -1;/*** 供应商名称*/public String name() default "";/*** 供应商地址*/public String address() default "";
}
FruitInfoUtil.java
import java.lang.reflect.Field;/*** 注解处理器*/
public class FruitInfoUtil {public static void getFruitInfo(Class<?> clazz){String strFruitName=" 水果名称:";String strFruitColor=" 水果颜色:";String strFruitProvicer="供应商信息:";Field[] fields = clazz.getDeclaredFields();for(Field field :fields){if(field.isAnnotationPresent(FruitName.class)){FruitName fruitName = (FruitName) field.getAnnotation(FruitName.class);strFruitName=strFruitName+fruitName.value();System.out.println(strFruitName);}else if(field.isAnnotationPresent(FruitColor.class)){FruitColor fruitColor= (FruitColor) field.getAnnotation(FruitColor.class);strFruitColor=strFruitColor+fruitColor.fruitColor().toString();System.out.println(strFruitColor);}else if(field.isAnnotationPresent(FruitProvider.class)){FruitProvider fruitProvider= (FruitProvider) field.getAnnotation(FruitProvider.class);strFruitProvicer=" 供应商编号:"+fruitProvider.id()+" 供应商名称:"+fruitProvider.name()+" 供应商地址:"+fruitProvider.address();System.out.println(strFruitProvicer);}}}
}
Apple.java
import test.FruitColor.Color;/*** 注解使用*/
public class Apple {@FruitName("Apple")private String appleName;@FruitColor(fruitColor=Color.RED)private String appleColor;@FruitProvider(id=1,name="陕西红富士集团",address="陕西省西安市延安路89号红富士大厦")private String appleProvider;public void setAppleColor(String appleColor) {this.appleColor = appleColor;}public String getAppleColor() {return appleColor;}public void setAppleName(String appleName) {this.appleName = appleName;}public String getAppleName() {return appleName;}public void setAppleProvider(String appleProvider) {this.appleProvider = appleProvider;}public String getAppleProvider() {return appleProvider;}public void displayName(){System.out.println("水果的名字是:苹果");}
}
FruitRun.java
/*** 输出结果*/
public class FruitRun {public static void main(String[] args) {FruitInfoUtil.getFruitInfo(Apple.class);}
}
运行结果是:
水果名称:Apple
水果颜色:RED
供应商编号:1 供应商名称:陕西红富士集团 供应商地址:陕西省西安市延安路89号红富士大厦
参考链接:
[1]http://www.cnblogs.com/peida/archive/2013/04/26/3038503.html
[2]http://www.cnblogs.com/whoislcj/p/5671622.html
[3]http://blog.csdn.net/lylwo317/article/details/52163304
热门内容:你说,一个Java字符串到底有多少个字符?
在Java项目中打印错误日志的正确姿势,排查问题更方便,非常实用!一款vue编写的功能强大的swagger-ui,有点秀(附开源地址)BeanUtils 是用 Spring 的还是 Apache 的好?因用了Insert into select语句,美女同事被开除了!最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡
你分析过@Annotation注解的实现原理吗?相关推荐
- Java反射学习总结五(Annotation(注解)-基础篇)
Annotation(注解)简单介绍: 注解大家印象最深刻的可能就是JUnit做单元測试,和各种框架里的使用了. 本文主要简介一下注解的用法,下篇文章再深入的研究. annotation并不直接影响代 ...
- @Autowired注解的实现原理
@Autowired注解用法 在分析这个注解的实现原理之前,我们不妨先来回顾一下@Autowired注解的用法. 将@Autowired注解应用于构造函数,如以下示例所示 public class M ...
- Spring框架你敢写精通,面试官就敢问@Autowired注解的实现原理
面试官:Spring框架中的@Autowired注解可以标注在哪些地方? 小小白:@Autowired注解可以被标注在构造函数.属性.setter方法或配置方法上,用于实现依赖自动注入. 面试官:有没 ...
- 声明属性Hibernate的Annotation注解
工作之余抽点时间出来写写博文,希望对新接触的朋友有帮助.今天在这里和大家一起学习一下声明属性 当项目变得比较大的时候,如何还应用hbm.xml文件来配置Hibernate实体就会变得比较复杂.这里Hi ...
- Java反射自定义注解底层设计原理
文章目录 一.反射 1. 反射概念 2. 反射机制的优缺点 3. 反射的用途 4. 反射技术的使用 5. 反射常用的Api 6. 反射执行构造函数 7. 反射执行给属性赋值 8. 反射执行调用方法 二 ...
- 从源码分析 Spring 基于注解的事务
从源码分析 Spring 基于注解的事务 在spring引入基于注解的事务(@Transactional)之前,我们一般都是如下这样进行拦截事务的配置: <!-- 拦截器方式配置事务 --> ...
- java 数据校验框架_自己写的基于java Annotation(注解)的数据校验框架
JavaEE6中提供了基于java Annotation(注解)的Bean校验框架,Hibernate也有类似的基于Annotation的数据校验功能,我在工作中,产品也经常需要使 用数据校验,为了方 ...
- java中注解动态传参_Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性)...
Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性)java 前言:因为前段时间忙于写接口,在接口中须要作不少的参数校验,本着简洁.高效的原则,便写了这个小工具供本身使 ...
- @Autowire注解的工作原理
@Autowired注解的工作原理 @Autowired注解用法 将@Autowired注解应用于构造函数,如以下示例所示 public class MovieRecommender {private ...
最新文章
- IJCV2021 人脸关键点检测器PIPNet
- 关于《红楼梦》的读后感优秀范文2000字
- Java数据结构和算法:HashMap的实现原理
- 学校计算机房的布线注意要点,校园网络布线实施中的注意事项
- 科大星云诗社动态20201122
- 技术圈儿007---Redis 生产架构选型解决方案
- 《中国人工智能学会通讯》——6.7 实体链接任务及系统
- i219v微星 驱动_适用于WinPE的I219V英特尔驱动程序
- 员工的12个需求及实现
- [译]C语言实现一个简易的Hash table(2)
- mybatis与data jpa
- yolov2-coco数据集网络架构
- 无线路由器桥接完整教程(不会断网)【图文详解】
- linux读取ads1115ADC例程
- 消除Permission is only granted to system apps报错
- Cisco交换机产品线和主要产品--- 型号说明
- 机器学习六步曲——“小马医生”养成记
- React Native 实践之携程 Moles 框架
- zhong yu gong si
- R语言笔记⑧——数据挖掘算法
热门文章
- Git远程仓库地址变更
- QWidget一生,从创建到销毁事件流
- Oracle总结第二篇【视图、索引、事务、用户权限、批量操作】
- list,set,map,数组间的相互转换
- 仲兆鹏 160809329 第5次
- 【转帖】SQLServer登录连接失败(error:40-无法打开到SQLServer的连接)的解决方案...
- Spring3.0 AOP 具体解释
- “System.Data.OracleClient.OracleConnection”已过时
- PHP中spl_autoload_register函数的用法
- FloodFill 图像分割