Many APIs require a fair amount of boilerplate code. For example, in order
to write a JAX-RPC web service, you must provide a paired interface and
implementation. This boilerplate could be generated automatically by a tool
if the program were “decorated” with annotations indicating which methods
were remotely accessible.
Other APIs require “side files” to be maintained in parallel with programs.
For example JavaBeans requires a
BeanInfo
class to be maintained in parallel
with a bean, and Enterprise JavaBeans (EJB) requires a deployment descriptor.
It would be more convenient and less error-prone if the information in these
side files were maintained as annotations in the program itself.
The Java platform has always had various ad hoc annotation mechanisms. For
example the
transient
modifier is an ad hoc annotation indicating that a field
should be ignored by the serialization subsystem, and the
@deprecated
javadoc
tag is an ad hoc annotation indicating that the method should no longer be used.
As of release 5.0, the platform has a general purpose annotation (also known
as metadata) facility that permits you to define and use your own annotation types.
The facility consists of a syntax for declaring annotation types, a syntax for
annotating declarations, APIs for reading annotations, a class file representation
for annotations, and an annotation processing tool.
Annotations do not directly affect program semantics, but they do affect the way
programs are treated by tools and libraries, which can in turn affect the semantics
of the running program. Annotations can be read from source files, class files, or
reflectively at run time.
Annotations complement javadoc tags. In general, if the markup is intended to affect
or produce documentation, it should probably be a javadoc tag; otherwise, it should
be an annotation.
Typical application programmers will never have to define an annotation type, but it
is not hard to do so. Annotation type declarations are similar to normal interface
declarations. An at-sign (
@
) precedes the
interface
keyword. Each method declaration
defines an element of the annotation type. Method declarations must not have any
parameters or a
throws
clause. Return types are restricted to primitives,
String
,
Class
, enums, annotations, and arrays of the preceding types. Methods can have
default values. Here is an example annotation type declaration:
/** * Describes the Request-For-Enhancement(RFE) that led * to the presence
of the annotated API element. */public @interface RequestForEnhancement {
int id();
String synopsis();
String engineer() default "[unassigned]";
String date(); default "[unimplemented]";
}
Once an annotation type is defined, you can use it to annotate declarations. An
annotation is a special kind of modifier, and can be used anywhere that other
modifiers (such as
public
,
static
, or
final
) can be used. By convention, annotations
precede other modifiers. Annotations consist of an at-sign (
@
) followed by an
annotation type and a parenthesized list of element-value pairs. The values must
be compile-time constants. Here is a method declaration with an annotation corresponding
to the annotation type declared above:
@RequestForEnhancement(
id = 2868724,
synopsis = "Enable time-travel",
engineer = "Mr. Peabody",
date = "4/1/3007")
public static void travelThroughTime(Date destination) { ... }
An annotation type with no elements is termed a
marker annotation type, for example:
/** * Indicates that the specification of the annotated API element * is
preliminary and subject to change. */public @interface Preliminary { }
It is permissible to omit the parentheses in marker annotations, as shown below:
@Preliminary public class TimeTravel { ... }
In annotations with a single element, the element should be named
value
, as shown below:
/**
* Associates a copyright notice with the annotated API element.
*/public @interface Copyright { String value();}
It is permissible to omit the element name and equals sign (
=
) in a single-element
annotation whose element name is
value
, as shown below:
@Copyright("2002 Yoyodyne Propulsion Systems")
public class OscillationOverthruster { ... }
To tie it all together, we'll build a simple annotation-based test framework. First
we need a marker annotation type to indicate that a method is a test method, and
should be run by the testing tool:
import java.lang.annotation.*;
/**
* Indicates that the annotated method is a test method.
* This annotation should be used only on parameterless static methods.
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Test { }
Note that the annotation type declaration is itself annotated. Such annotations are
called meta-annotations. The first (
@Retention(RetentionPolicy.RUNTIME)
) indicates
that annotations with this type are to be retained by the VM so they can be read
reflectively at run-time. The second (
@Target(ElementType.METHOD)
) indicates that
this annotation type can be used to annotate only method declarations.
Here is a sample program, some of whose methods are annotated with the above interface:
public class Foo {
@Test
public static void m1() { }
public static void m2() { }
@Test
public static void m3() {
throw new RuntimeException("Boom");
}
public static void m4() { }
@Test
public static void m5() { }
public static void m6() { }
@Test
public static void m7() {
throw new RuntimeException("Crash");
}
public static void m8() { }}
Here is the testing tool:
import java.lang.reflect.*;
public class RunTests {
public static void main(String[] args) throws Exception {
int passed = 0, failed = 0;
for (Method m : Class.forName(args[0]).getMethods()) {
if (m.isAnnotationPresent(Test.class)) {
try {
m.invoke(null);
passed++;
} catch (Throwable ex) {
System.out.printf("Test %s failed: %s %n", m, ex.getCause());
failed++;
}
}
}
System.out.printf("Passed: %d, Failed %d%n", passed, failed); }}
The tool takes a class name as a command line argument and iterates over all the
methods of the named class attempting to invoke each method that is annotated
with the
Test
annotation type (defined above). The reflective query to find out
if a method has a
Test
annotation is highlighted in green. If a test method
invocation throws an exception, the test is deemed to have failed, and a failure
report is printed. Finally, a summary is printed showing the number of tests that
passed and failed. Here is how it looks when you run the testing tool on the
Foo
program (above):
$ java RunTests FooTest public static void Foo.m3() failed:
java.lang.RuntimeException: Boom Test public static void Foo.m7() failed:
java.lang.RuntimeException: Crash Passed: 2, Failed 2
While this testing tool is clearly a toy, it demonstrates the power of annotations
and could easily be extended to overcome its limitations.
相关推荐
赠送jar包:annotations-13.0.jar; 赠送原API文档:annotations-13.0-javadoc.jar; 赠送源代码:annotations-13.0-sources.jar; 赠送Maven依赖信息文件:annotations-13.0.pom; 包含翻译后的API文档:...
赠送jar包:annotations-23.0.0.jar; 赠送原API文档:annotations-23.0.0-javadoc.jar; 赠送源代码:annotations-23.0.0-sources.jar; 赠送Maven依赖信息文件:annotations-23.0.0.pom; 包含翻译后的API文档:...
赠送jar包:swagger-annotations-1.5.20.jar; 赠送原API文档:swagger-annotations-1.5.20-javadoc.jar; 赠送源代码:swagger-annotations-1.5.20-sources.jar; 赠送Maven依赖信息文件:swagger-annotations-...
赠送jar包:swagger-annotations-2.1.2.jar; 赠送原API文档:swagger-annotations-2.1.2-javadoc.jar; 赠送源代码:swagger-annotations-2.1.2-sources.jar; 赠送Maven依赖信息文件:swagger-annotations-2.1.2....
赠送jar包:swagger-annotations-1.6.2.jar; 赠送原API文档:swagger-annotations-1.6.2-javadoc.jar; 赠送源代码:swagger-annotations-1.6.2-sources.jar; 赠送Maven依赖信息文件:swagger-annotations-1.6.2....
赠送jar包:audience-annotations-0.5.0.jar; 赠送原API文档:audience-annotations-0.5.0-javadoc.jar; 赠送源代码:audience-annotations-0.5.0-sources.jar; 赠送Maven依赖信息文件:audience-annotations-...
赠送jar包:jackson-annotations-2.9.0.jar; 赠送原API文档:jackson-annotations-2.9.0-javadoc.jar; 赠送源代码:jackson-annotations-2.9.0-sources.jar; 赠送Maven依赖信息文件:jackson-annotations-2.9.0....
赠送jar包:j2objc-annotations-1.3.jar; 赠送原API文档:j2objc-annotations-1.3-javadoc.jar; 赠送源代码:j2objc-annotations-1.3-sources.jar; 赠送Maven依赖信息文件:j2objc-annotations-1.3.pom; 包含...
赠送jar包:swagger-annotations-1.5.24.jar; 赠送原API文档:swagger-annotations-1.5.24-javadoc.jar; 赠送源代码:swagger-annotations-1.5.24-sources.jar; 赠送Maven依赖信息文件:swagger-annotations-...
赠送jar包:jackson-annotations-2.13.1.jar; 赠送原API文档:jackson-annotations-2.13.1-javadoc.jar; 赠送源代码:jackson-annotations-2.13.1-sources.jar; 赠送Maven依赖信息文件:jackson-annotations-...
赠送jar包:error_prone_annotations-2.3.2.jar; 赠送原API文档:error_prone_annotations-2.3.2-javadoc.jar; 赠送源代码:error_prone_annotations-2.3.2-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:error_prone_annotations-2.10.0.jar; 赠送原API文档:error_prone_annotations-2.10.0-javadoc.jar; 赠送源代码:error_prone_annotations-2.10.0-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:jackson-annotations-2.11.4.jar; 赠送原API文档:jackson-annotations-2.11.4-javadoc.jar; 赠送源代码:jackson-annotations-2.11.4-sources.jar; 赠送Maven依赖信息文件:jackson-annotations-...
赠送jar包:j2objc-annotations-1.1.jar; 赠送原API文档:j2objc-annotations-1.1-javadoc.jar; 赠送源代码:j2objc-annotations-1.1-sources.jar; 赠送Maven依赖信息文件:j2objc-annotations-1.1.pom; 包含...
赠送jar包:error_prone_annotations-2.5.1.jar; 赠送原API文档:error_prone_annotations-2.5.1-javadoc.jar; 赠送源代码:error_prone_annotations-2.5.1-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:error_prone_annotations-2.2.0.jar; 赠送原API文档:error_prone_annotations-2.2.0-javadoc.jar; 赠送源代码:error_prone_annotations-2.2.0-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:error_prone_annotations-2.0.18.jar; 赠送原API文档:error_prone_annotations-2.0.18-javadoc.jar; 赠送源代码:error_prone_annotations-2.0.18-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:error_prone_annotations-2.3.4.jar; 赠送原API文档:error_prone_annotations-2.3.4-javadoc.jar; 赠送源代码:error_prone_annotations-2.3.4-sources.jar; 赠送Maven依赖信息文件:error_prone_...
赠送jar包:jackson-module-jaxb-annotations-2.7.8.jar; 赠送原API文档:jackson-module-jaxb-annotations-2.7.8-javadoc.jar; 赠送源代码:jackson-module-jaxb-annotations-2.7.8-sources.jar; 赠送Maven依赖...
赠送jar包:error_prone_annotations-2.1.3.jar; 赠送原API文档:error_prone_annotations-2.1.3-javadoc.jar; 赠送源代码:error_prone_annotations-2.1.3-sources.jar; 赠送Maven依赖信息文件:error_prone_...