style="text-indent:2em;">今天给各位分享Java的注解是什么原理的知识,其中也会对java 注释 不建议使用进行解释,如果能碰巧解决你现在面临的问题,别忘了关注本站,现在开始吧!
本文目录
Spring boot是Java程序员必须掌握的框架之一么
毫不夸张的说:是的。
目前,不管是个人开发项目还是企业开发项目,如果选择的是Java语言,那我们第一个想到的就是用SpringBoot框架开发,因为用它太快太爽了。几分钟就可以将一个Web后端的项目跑起来。而我们只需要写业务代码就好了。这对于程序员来说简直就是天大的福利。
想想我们用SpringMVC的时候,是多么的痛苦。虽然SpringMVC也是非常优秀的框架(这也是必须要掌握的Java框架),但是基于XML文件的配置实在是太繁琐。启动一个JavaWeb项目,我们需要写一大堆的xml文件,spring.xml、application.xml、web.xml、springmvc-servlet.xml等等,不胜其烦。而且即便这些东西都配置好了,项目还不一定能启动起来。很多程序员应该都碰到过这个问题,bean加载的问题、数据源配置的问题、注解扫描路劲的问题,等等。反正,即使再NB的程序员,没个半个一个小时,跑步起来基于SpringMVC框架的项目。(但是这不影响它成为一个非常优秀的Java框架)。
而SpringBoot完全抛弃了繁琐的xml,在SpringBoot的项目中,几乎看不到xml文件。干净清爽的配置让人欲罢不能,这也是为什么SpringBoot目前这么火的原因,也是现阶段,Java程序员必须要掌握的框架。
最后总结一句:SpringBoot的好,谁用谁知道。怎样理解Java的注解并运用注解编程
一、什么是注解
注解也叫元数据,例如我们常见的@Override和@Deprecated,注解是JDK1.5版本开始引入的一个特性,
用于对代码进行说明,可以对包、类、接口、字段、方法参数、局部变量等进行注解。
一般常用的注解可以分为三类:
1、Java自带的标准注解,包括@Override(标明重写某个方法)、@Deprecated(标明某个类或方法过时)和@SuppressWarnings(标明要忽略的警告),使用这些注解后编译器就会进行检查。
2、类为元注解,元注解是用于定义注解的注解,包括@Retention(标明注解被保留的阶段)、@Target(标明注解使用的范围)、@Inherited(标明注解可继承)、@Documented(标明是否生成javadoc文档)
3、类为自定义注解,可以根据自己的需求定义注解
二、注解的用途
在看注解的用途之前,有必要简单的介绍下XML和注解区别,
注解:是一种分散式的元数据,与源代码紧绑定。
xml:是一种集中式的元数据,与源代码无绑定
当然网上存在各种XML与注解的辩论哪个更好,这里不作评论和介绍,主要介绍一下注解的主要用途:
生成文档,通过代码里标识的元数据生成javadoc文档。
编译检查,通过代码里标识的元数据让编译器在编译期间进行检查验证。
编译时动态处理,编译时通过代码里标识的元数据动态处理,例如动态生成代码。
运行时动态处理,运行时通过代码里标识的元数据动态处理,例如使用反射注入实例
三、注解的实现原理
实现注解三要素:
1、注解声明
2、使用注解的元素
3、操作注解使其起作用(注解处理器)
注解声明
首先我们让看一下java中的元注解(也就是上面提到的注解的注解),总共有4个如下:
@Target,@Retention,@Documented,@Inherited这4个元注解都是在jdk的java.lang.annotation包下面,
@Target:Target说明的是Annotation所修饰的对象范围。@Retention:
定义了该Annotation被保留的时间长短:某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略,
而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。
使用这个元注解可以对Annotation的“生命周期”限制。@Documented:@Documented用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员
@Inherited:@Inherited元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,
则这个annotation将被用于该class的子类。
注意:@Inheritedannotation类型是被标注过的class的子类所继承。类并不从它所实现的接口继承annotation,方法并不从它所重载的方法继承annotation。
当@Inheritedannotation类型标注的annotation的Retention是RetentionPolicy.RUNTIME,则反射API增强了这种继承性。如果我们使用java.lang.reflect去查询一个@Inheritedannotation类型的annotation时,
反射代码检查将展开工作:检查class和其父类,直到发现指定的annotation类型被发现,或者到达类继承结构的顶层。
java.lang.reflect.AnnotatedElement接口是所有程序元素(Class、Method和Constructor)的父接口,所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下四个个方法来访问Annotation信息:
方法1:<TextendsAnnotation>TgetAnnotation(Class<T>annotationClass):返回改程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。
方法2:Annotation[]getAnnotations():返回该程序元素上存在的所有注解。
方法3:booleanisAnnotationPresent(Class<?extendsAnnotation>annotationClass):判断该程序元素上是否
target注解作用
Target作用于修饰的注释可以修饰的类型范围。
target英文有目标、目的的意思。@Target在java中是注释类。@Target作用于修饰的注释可以修饰的类型范围。
Target包含一个ElementType[]元素类型的数组。ElementType[]数组值value,表明Target修饰的注释可以修饰的类型范围。ElementType枚举值包含方法、属性、类等等。
Java的注解是什么原理
首先,从一个简单的注解开始,@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public@interfaceTest{
Stringvalue()default"";
}
使用javap-verbose命令查看这个注解的class文件,发现这个注解被编译成了接口,并且继承了java.lang.annotation.Annotation接口,接口是不能直接实例化使用的,当在代码中使用这个注解,并使用getAnnotation方法获取注解信息时,JVM通过动态代理的方式生成一个实现了Test接口的代理对象实例,然后对该实例的属性赋值,value值就存在这个代理对象实例中。
如果顺着getAnnotation方法继续跟踪源代码,会发现创建代理对象是在AnnotationParser.java中实现的,这个类中有一个annotationForMap方法。在annotationForMap方法内部使用Proxy.newProxyInstance方法在运行时动态创建代理,AnnotationInvocationHandler实现了InvocationHandler接口,当调用代理对象的value()方法获取注解的value值,就会进入AnnotationInvocationHandler类中的invoke方法,深入invoke方法会发现,获取value值最终是从AnnotationInvocationHandler类的memberValues属性中获取的,memberValues是一个Map类型,key是注解的属性名,这里就是“value”,value是使用注解时设置的值。
因为回答中不能插入源码做讲解辅助,更多详细内容可以关注『Java实战技术』主页的原创文章《Java注解是如何玩转的,面试官和我聊了半个小时》。
好了,文章到此结束,希望可以帮助到大家。