ImageVerifierCode 换一换
格式:DOCX , 页数:11 ,大小:24.40KB ,
资源ID:7078988      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/7078988.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(Java Annotation手册.docx)为本站会员(b****6)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

Java Annotation手册.docx

1、Java Annotation手册Java annotation 手册前言:在上篇文章Java Annotation入门中概要性的介绍了Annotation的定义、使用,范围涵盖较广,但是深度不够。所以作者在Java Annotation入门后,继续整理了Annotation的概念和知识点,与喜欢research的朋友们共享。阅读提示:文中提到的程序成员或者程序元素是一个概念,指组成程序代码的单元:如类、方法、成员变量。一、Annotation究竟是什么?Annotation提供了一条与程序元素关联任何信息或者任何元数据(metadata)的途径。从某些方面看,annotation就像修饰符一

2、样被使用,并应用于包、类型、构造方法、方法、成员变量、参数、本地变量的声明中。这些信息被存储在annotation的“name=value”结构对中。annotation类型是一种接口,能够通过java反射API的方式提供对其信息的访问。annotation能被用来为某个程序元素(类、方法、成员变量等)关联任何的信息。需要注意的是,这里存在着一个基本的潜规则:annotaion不能影响程序代码的执行,无论增加、删除annotation,代码都始终如一的执行。另外,尽管一些annotation通过java的反射api方法在运行时被访问,而java语言解释器在工作时忽略了这些annotation。

3、正是由于java虚拟机忽略了annotation,导致了annotation类型在代码中是“不起作用”的;只有通过某种配套的工具才会对annotation类型中的信息进行访问和处理。本文中将涵盖标准的annotation和meta-annotation类型,陪伴这些annotation类型的工具是java编译器(当然要以某种特殊的方式处理它们)。由于上述原因,annotation在使用时十分简便。一个本地变量可以被一个以NonNull命名的annotation类型所标注,来作为对这个本地变量不能被赋予null值的断言。而我们可以编写与之配套的一个annotation代码分析工具,使用它来对具有

4、前面变量的代码进行解析,并且尝试验证这个断言。当然这些代码并不必自己编写。在JDK安装后,在JDK/bin目录中可以找到名为“apt”的工具,它提供了处理annotation的框架:它启动后扫描源代码中的annotation,并调用我们定义好的annotation处理器完成我们所要完成的工作(比如验证前面例子中的断言)。说到这里,annotation的强大功能似乎可以替代XDoclet这类的工具了,随着我们的深入,大家会更加坚信这一点。注:详细描述请参看jsr250规范:http:/www.jcp.org/aboutJava/communityprocess/pfd/jsr250/二、Anno

5、tation的定义:这段文字开始介绍annotation相关技术。在此大家将看到java5.0的标准annotation类型,这种标准类型就是前文中所说的“内建”类型,它们可以直接被javac支持。可喜的是,在java6.0beta版中的javac已经加入了对自定义annotation的支持。1。Annotation的概念和语法:首先,关键的概念是理解annotation是与一个程序元素相关联信息或者元数据的标注。它从不影响java程序的执行,但是对例如编译器警告或者像文档生成器等辅助工具产生影响。下面是常用的annotation列表,我们应该注意在annotation和annotation类

6、型之间的不同:A.annotation:annotation使用了在java5.0所带来的新语法,它的行为十分类似public、final这样的修饰符。每个annotation具有一个名字和成员个数=0。每个annotation的成员具有被称为name=value对的名字和值(就像javabean一样),name=value装载了annotation的信息。B.annotation类型:annotation类型定义了annotation的名字、类型、成员默认值。一个annotation类型可以说是一个特殊的java接口,它的成员变量是受限制的,而声明annotation类型时需要使用新语法。当

7、我们通过java反射api访问annotation时,返回值将是一个实现了该annotation类型接口的对象,通过访问这个对象我们能方便的访问到其annotation成员。后面的章节将提到在java5.0的java.lang包里包含的3个标准annotation类型。C.annotation成员:annotation的成员在annotation类型中以无参数的方法的形式被声明。其方法名和返回值定义了该成员的名字和类型。在此有一个特定的默认语法:允许声明任何annotation成员的默认值:一个annotation可以将name=value对作为没有定义默认值的annotation成员的值,当

8、然也可以使用name=value对来覆盖其它成员默认值。这一点有些近似类的继承特性,父类的构造函数可以作为子类的默认构造函数,但是也可以被子类覆盖。D.marker annotation类型:一个没有成员定义的annotation类型被称为marker annotation。这种annotation类型仅使用自身的存在与否来为我们提供信息。如后面要说的Override。E.meta-annotation:meta-annotation也称为元annotation,它是被用来声明annotation类型的annotation。Java5.0提供了一些标准的元-annotation类型。下面介绍的

9、target、retention就是meta-annotation。F.target:annotation的target是一个被标注的程序元素。target说明了annotation所修饰的对象范围:annotation可被用于packages、types(类、接口、枚举、annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在annotation类型的声明中使用了target可更加明晰其修饰的目标。G.retention:annotation的retention定义了该annotation被保留的时间长短:某些annota

10、tion仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为annotation与class在使用上是被分离的)。使用这个meta-annotation可以对annotation的“生命周期”限制。H.metadata:由于metadata被广泛使用于各种计算机开发过程中,所以当我们在这里谈论的metadata即元数据通常指被annotation装载的信息或者annotation本身。2。使用标准Annotation:java5.0在jav

11、a.lang包中定义了3种标准的annotation类型:A.Override:java.lang.Override是一个marker annotation类型,它被用作标注方法。它说明了被标注的方法重载了父类的方法,起到了断言的作用。如果我们使用了这种annotation在一个没有覆盖父类方法的方法时,java编译器将以一个编译错误来警示。这个annotaton常常在我们试图覆盖父类方法而确又写错了方法名时发挥威力。使用方法极其简单:在使用此annotation时只要在被修饰的方法前面加上Override。下面的代码是一个使用Override修饰一个企图重载父类的toString方法,而又存

12、在拼写错误的sample:Overridepublic String toSting() / 注意方法名拼写错了return + super.toString() + ;B.Deprecated:同样Deprecated也是一个marker annotation。当一个类型或者类型成员使用Deprecated修饰的话,编译器将不鼓励使用这个被标注的程序元素。而且这种修饰具有一定的“延续性”:如果我们在代码中通过继承或者覆盖的方式使用了这个过时的类型或者成员,虽然继承或者覆盖后的类型或者成员并不是被声明为Deprecated,但编译器仍然要报警。值得注意,Deprecated这个annotati

13、on类型和javadoc中的deprecated这个tag是有区别的:前者是java编译器识别的,而后者是被javadoc工具所识别用来生成文档(包含程序成员为什么已经过时、它应当如何被禁止或者替代的描述)。在java5.0,java编译器仍然象其从前版本那样寻找deprecated这个javadoc tag,并使用它们产生警告信息。但是这种状况将在后续版本中改变,我们应在现在就开始使用Deprecated来修饰过时的方法而不是deprecated javadoc tag。下面是一段使用Deprecated的代码:/* * 这里是javadoc的deprecated声明. * deprecat

14、ed No one has players for this format any more.Use VHS instead. */Deprecated public class Betamax . C.SuppressWarnings:SuppressWarnings被用于有选择的关闭编译器对类、方法、成员变量、变量初始化的警告。在java5.0,sun提供的javac编译器为我们提供了-Xlint选项来使编译器对合法的程序代码提出警告,此种警告从某种程度上代表了程序错误。例如当我们使用一个generic collection类而又没有提供它的类型时,编译器将提示出unchecked war

15、ning的警告。通常当这种情况发生时,我们就需要查找引起警告的代码。如果它真的表示错误,我们就需要纠正它。例如如果警告信息表明我们代码中的switch语句没有覆盖所有可能的case,那么我们就应增加一个默认的case来避免这种警告。相仿,有时我们无法避免这种警告,例如,我们使用必须和非generic的旧代码交互的generic collection类时,我们不能避免这个unchecked warning。此时SuppressWarning就要派上用场了,在调用的方法前增加SuppressWarnings修饰,告诉编译器停止对此方法的警告。SuppressWarning不是一个marker an

16、notation。它有一个类型为String的成员,这个成员的值为被禁止的警告名。对于javac编译器来讲,被-Xlint选项有效的警告名也同样对SuppressWarings有效,同时编译器忽略掉无法识别的警告名。annotation语法允许在annotation名后跟括号,括号中是使用逗号分割的name=value对用于为annotation的成员赋值:SuppressWarnings(value=unchecked,fallthrough)public void lintTrap() /* sloppy method body omitted */ 在这个例子中SuppressWarni

17、ngs annotation类型只定义了一个单一的成员,所以只有一个简单的value=.作为name=value对。又由于成员值是一个数组,故使用大括号来声明数组值。注意:我们可以在下面的情况中缩写annotation:当annotation只有单一成员,并成员命名为value=。这时可以省去value=。比如将上面的SuppressWarnings annotation进行缩写:SuppressWarnings(unchecked,fallthrough)如果SuppressWarnings所声明的被禁止警告个数为一个时,可以省去大括号:SuppressWarnings(unchecked)

18、3。Annotation语法:在上一个章节中,我们看到书写marker annotation和单一成员annotation的语法。下面本人来介绍一下完整的语法:annotation由“+annotation类型名称+(.逗号分割的name-value对.)”组成。其中成员可以按照任何的顺序。如果annotation类型定义了某个成员的默认值,则这个成员可以被省略。成员值必须为编译时常量、内嵌的annotation或者数组。下面我们将定义一个annotation类型名为Reviews,它有一个由Review annotation数组构成的成员。这个Review annotation类型有三个成员

19、:reviewer是一个字符串,comment 是一个具有默认值的可选的字符串,grade是一个Review.Grade枚举类型值。Reviews(/ Single-value annotation, so value= is omitted hereReview(grade=Review.Grade.EXCELLENT,reviewer=df),Review(grade=Review.Grade.UNSATISFACTORY,reviewer=eg,comment=This method needs an Override annotation)annotation语法的另一个重要规则是没有

20、程序成员可以有多于一个的同一annotation实例。例如在一个类中简单的放置多个Review annotation。这也是在上面代码中定义Reviews annotation类型数组的原因。4。Annotation成员类型和值:annotation成员必须是非空的编译时常量表达式。可用的成员类型为:primitive类型、, String, Class, enumerated类型, annotation类型, 和前面类型的数组。下面我们定义了一个名为UncheckedExceptions 的annotation类型,它的成员是一个扩展了RuntimeException类的类数组。Unchec

21、kedExceptions(IllegalArgumentException.class, StringIndexOutOfBoundsException.class)5。Annotation的目标:annotation通常被放在类型定义和成员定义的前面。然而它也出现在package、方法参数、本地变量的前面。下面,我们来讨论一下这些不大常用的写法:package annotation出现在package声明的前面。下面的例子package-info.java中不包含任何的公共类型定义,却包含一个可选的javadoc注释。/* * This package holds my custom an

22、notation types. */com.davidflanagan.annotations.Author(David Flanagan)package com.davidflanagan.annotations;当package-info.java文件被编译时,它将产生名为包含annotation(特殊的接口)声明的package-info.class的类。这个接口没有成员,它的名字package-info不是一个合法的java标识,所以它不能用在java源代码中。这个接口的存在只是简单的被看作一个为package annotation准备的占位符。用于修饰方法参数、catch参数、本地变

23、量的annotation只是简单的出现在这些程序成员的修饰符位置。java类文件格式没有为本地变量或者catch参数存储annotation作准备,所以这些annotation总是保留在源代码级别(source retention);方法参数annotation能够保存在类文件中,也可以在保留到运行时。最后,请注意,枚举类型定义中不允许任何的修饰符修饰其枚举值。6。Annotation和默认值:在Annotation中,没有默认值的成员必须有一个成员值。而如何理解默认值是如何被处理就是一个很重要的细节:annotation类型所定义的成员默认值被存储在class文件中,不被编译到annotat

24、ion里面。如果我们修改一个annotation类型使其成员的默认值发生了改变,这个改变对于所有此类型的annotation中没有明确提供成员值的成员产生影响(即修改了该成员的成员值)。即使在annotation类型使其成员的默认值被改变后annotation从没被重新编译过,该类型的annotation(改变前已经被编译的)也受到影响。三、Annotation工作原理:Annotation与反射在java5.0中Java.lang.reflect提供的反射API被扩充了读取运行时annotation的能力。让我们回顾一下前面所讲的:一个annotation类型被定义为runtime rete

25、ntion后,它才是在运行时可见,当class文件被装载时被保存在class文件中的annotation才会被虚拟机读取。那么reflect是如何帮助我们访问class中的annotation呢?下文将在java.lang.reflect用于annotation的新特性,其中java.lang.reflect.AnnotatedElement是重要的接口,它代表了提供查询annotation能力的程序成员。这个接口被java.lang.Package、java.lang.Class实现,并间接地被Method类、Constructor类、java.lang.reflect的Field类实现。而

26、annotation中的方法参数可以通过Method类、Constructor类的getParameterAnnotations()方法获得。下面的代码使用了AnnotatedElement类的isAnnotationPresent()方法判断某个方法是否具有Unstable annotation,从而断言此方法是否稳定:import java.lang.reflect.*;Class c = WhizzBangClass.class; Method m = c.getMethod(whizzy, int.class, int.class);boolean unstable = m.isAnn

27、otationPresent(Unstable.class);isAnnotationPresent()方法对于检查marker annotation是十分有用的,因为marker annotation没有成员变量,所以我们只要知道class的方法是否使用了annotation修饰就可以了。而当处理具有成员的annotation时,我们通过使用getAnnotation()方法来获得annotation的成员信息(成员名称、成员值)。这里我们看到了一套优美的java annotation系统:如果annotation存在,那么实现了相应的annotation类型接口的对象将被getAnnota

28、tion()方法返回,接着调用定义在annotation类型中的成员方法可以方便地获得任何成员值。回想一下,前面介绍的Reviews annotation,如果这个annotation类型被声明为runtime retention的话,我们通过下面的代码来访问Reviews annotation的成员值:AnnotatedElement target = WhizzBangClass.class; /获得被查询的AnnotatedElement/ 查询AnnotatedElement的Reviews annotation信息Reviews annotation = target.getAnno

29、tation(Reviews.class);/ 因为Reviews annotation类型的成员为Review annotation类型的数组,/ 所以下面声明了Review reviews保存Reviews annotation类型的value成员值。Review reviews = annotation.value();/ 查询每个Review annotation的成员信息for(Review r : reviews) Review.Grade grade = r.grade();String reviewer = r.reviewer();String comment = ment(

30、);System.out.printf(%s assigned a grade of %s and comment %s%n,reviewer, grade, comment);四、如何自定义Annotation?1详解annotation与接口的异同:因为annotation类型是一个非凡的接口,所以两者之间存在着某些差异:A.Annotation类型使用关键字interface而不是interface。这个关键字声明隐含了一个信息:它是继承了java.lang.annotation.Annotation接口,并非声明了一个interface。B.Annotation类型、方法定义是独特的、

31、受限制的。Annotation类型的方法必须声明为无参数、无异常抛出的。这些方法定义了annotation的成员:方法名成为了成员名,而方法返回值成为了成员的类型。而方法返回值类型必须为primitive类型、Class类型、枚举类型、annotation类型或者由前面类型之一作为元素的一维数组。方法的后面可以使用default和一个默认数值来声明成员的默认值,null不能作为成员默认值,这与我们在非annotation类型中定义方法有很大不同。Annotation类型和它的方法不能使用annotation类型的参数、成员不能是generic。只有返回值类型是Class的方法可以在annotation类型中使用gener

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1