编者的话:注解是java的一个主要特性且每个java开发者都应该知道如何使用它。

现在,是时候汇总这些和注解相关的信息到一篇文章了,祝大家阅读愉快。

目录

什么是注解

介绍

消费器

注解语法和注解元素

在什么地方使用

使用案例

内建注解

Java 8 与注解

自定义注解

提取注解

注解集成

使用注解的知名类库

小结

下载

资料

在这篇文章中我们将阐述什么是Java注解,它们如何工作,怎么使用它们。

我们将揭开Java注解的面纱,包括内建注解或称元注解,还将讨论Java8中与之相关的的新特性。

最后,我们将实现自定义的注解,编写一个使用注解的处理程序(消费器),它通过java反射使用注解。

我们还会列出一些基于注解,知名且被广泛应用的第三方类库如:Junit,JAXB,Spring,Hibernate。

在文章的最后,会有一个压缩文件包含了文章中的所有示例,实现这些例子使用的软件版本如下所示:

Eclipse Luna 4.4

JRE Update 8.20

Junit 4

Hibernate 4.3.6

FindBugs 3.0.0

1.什么是注解?

注解早在J2SE1.5就被引入到Java中,主要提供一种机制,这种机制允许程序员在编写代码的同时可以直接编写元数据。

在引入注解之前,程序员们描述其代码的形式尚未标准化,每个人的做法各异:transient关键字、注释、接口等。这显然不是一种优雅的方式,随之而来的一种崭新的记录元数据的形式——注解被引入到Java中。

其它因素也促成了这个决定:当时不同类型的应用程序使用XML作为标准的代码配置机制,这其实并不是最佳方式,因为代码和XML的解耦以及未来对这种解耦应用的维护并不低廉。另外,由于非保留字的使用,例如“@deprecated”自从Java1.4便开始在Java文档中使用。我非常确定这是一个现在在注解中使用“@”原因。

包含注解的设计和开发的Java规范主要有以下两篇:

2. 介绍

解释何为注解的最佳方式就是元数据这个词:描述数据自身的数据。注解就是代码的元数据,他们包含了代码自身的信息。

注解可以被用在包,类,方法,变量,参数上。自Java8起,有一种注解几乎可以被放在代码的任何位置,叫做类型注解。我们将会在后面谈到具体用法。

被注解的代码并不会直接被注解影响。这只会向第三系统提供关于自己的信息以用于不同的需求。

注解会被编译至class文件中,而且会在运行时被处理程序提取出来用于业务逻辑。当然,创建在运行时不可用的注解也是可能的,甚至可以创建只在源文件中可用,在编译时不可用的注解。

3.消费器

理解注解的目的以及如何使用它都会带来困难,因为注解本身并不包含任何功能逻辑,它们也不会影响自己注解的代码,那么,它们到底为什么而存在呢?

这个问题的解释就是我所称的注解消费器。它们是利用被注解代码并根据注解信息产生不同行为的系统或者应用程序。

例如,在Java自带的内建注解(元注解)中,消费器是执行被注解代码的JVM。还有其他稍后谈到的其他例子,例如JUnit,消费器是读取,分析被注解代码的JUnit处理程序,它还可以决定测试单元和方法执行顺序。我们会在JUnit章节更深入。

消费器使用Java中的反射机制来读取和分析被注解的源代码。使用的主要的包有:java.lang, java.lang.reflect。我们将会在本篇指南中介绍如何用反射从头开始创建一个自定义的消费器。

4. 注解语法和元素

声明一个注解需要使用“@”作为前缀,这便向编译器说明,该元素为注解。例如:

1

2

3

4@Annotation

public void annotatedMehod() {

...

}

上述的注解名称为Annotation,它正在注解annotatedMethod方法。编译器会处理它。注解可以以键值对的形式持有有很多元素,即注解的属性。

1

2

3

4

5

6

7@Annotation(

info ="I am an annotation",

counter ="55"

)

public void annotatedMehod() {

...

}

如果注解只包含一个元素(或者只需要指定一个元素的值,其它则使用默认值),可以像这样声明:

1

2

3

4@Annotation("I am an annotation")

public void annotatedMehod() {

...

}

就像我们看到的一样,如果没有元素需要被指定,则不需要括号。多个注解可以使用在同一代码上,例如类:

1

2

3@ Annotation (info ="U a u O")

@ Annotation2

class AnnotatedClass { ... }

一些java本身提供的开箱即用的注解,我们称之为内建注解。也可以定义你自己的注解,称之为子定义注解。我们会在下一章讨论。

5. 在什么地方使用

注解基本上可以在Java程序的每一个元素上使用:类,域,方法,包,变量,等等。

自Java8,诞生了通过类型注解的理念。在此之前,注解是限于在前面讨论的元素的声明上使用。从此,无论是类型还是声明都可以使用注解,就像:

1@MyAnnotation String str ="danibuiza";

我们将会在Java8关联章节看到这种机制的更多细节。

6. 使用案例

注解可以满足许多要求,最普遍的是:

向编译器提供信息:注解可以被编译器用来根据不同的规则产生警告,甚至错误。一个例子是Java8中@FunctionalInterface注解,这个注解使得编译器校验被注解的类,检查它是否是一个正确的函数式接口。

文档:注解可以被软件应用程序计算代码的质量例如:FindBugs,PMD或者自动生成报告,例如:用来Jenkins, Jira,Teamcity。

代码生成:注解可以使用代码中展现的元数据信息来自动生成代码或者XML文件,一个不错的例子是JAXB。

运行时处理:在运行时检查的注解可以用做不同的目的,像单元测试(JUnit),依赖注入(Spring),校验,日志(Log4j),数据访问(Hibernate)等等。

在这篇手册中我们将展现几种注解可能的用法,包括流行的Java类库是如何使用它们的。

7. 内建注解

Java语言自带了一系列的注解。在本章中我们将阐述最重要的一部分。这个清单只涉及了Java语言最核心的包,未包含标准JRE中所有包和库如JAXB或Servlet规范。

以下讨论到的注解中有一些被称之为Meta注解,它们的目的注解其他注解,并且包含关于其它注解的信息。

@Retention:这个注解注在其他注解上,并用来说明如何存储已被标记的注解。这是一种元注解,用来标记注解并提供注解的信息。可能的值是:

SOURCE:表明这个注解会被编译器忽略,并只会保留在源代码中。

CLASS:表明这个注解会通过编译驻留在CLASS文件,但会被JVM在运行时忽略,正因为如此,其在运行时不可见。

RUNTIME:表示这个注解会被JVM获取,并在运行时通过反射获取。

我们会在稍后展开几个例子。

@Target:这个注解用于限制某个元素可以被注解的类型。例如:

ANNOTATION_TYPE 表示该注解可以应用到其他注解上

CONSTRUCTOR 表示可以使用到构造器上

FIELD 表示可以使用到域或属性上

LOCAL_VARIABLE表示可以使用到局部变量上。

METHOD可以使用到方法级别的注解上。

PACKAGE可以使用到包声明上。

PARAMETER可以使用到方法的参数上

TYPE可以使用到一个类的任何元素上。

@Documented:被注解的元素将会作为Javadoc产生的文档中的内容。注解都默认不会成为成为文档中的内容。这个注解可以对其它注解使用。

@Inherited:在默认情况下,注解不会被子类继承。被此注解标记的注解会被所有子类继承。这个注解可以对类使用。

@Deprecated:说明被标记的元素不应该再度使用。这个注解会让编译器产生警告消息。可以使用到方法,类和域上。相应的解释和原因,包括另一个可取代的方法应该同时和这个注解使用。

@SuppressWarnings:说明编译器不会针对指定的一个或多个原因产生警告。例如:如果我们不想因为存在尚未使用的私有方法而得到警告可以这样做:

1

2

3

4@SuppressWarnings("unused")

private String myNotUsedMethod(){

...

}

通常,编译器会因为没调用该方而产生警告; 用了注解抑制了这种行为。该注解需要一个或多个参数来指定抑制的警告类型。

@Override:向编译器说明被注解元素是重写的父类的一个元素。在重写父类元素的时候此注解并非强制性的,不过可以在重写错误时帮助编译器产生错误以提醒我们。比如子类方法的参数和父类不匹配,或返回值类型不同。

@SafeVarargs:断言方法或者构造器的代码不会对参数进行不安全的操作。在Java的后续版本中,使用这个注解时将会令编译器产生一个错误在编译期间防止潜在的不安全操作。

8. Java 8 与注解

Java8带来了一些优势,同样注解框架的能力也得到了提升。在本章我们将会阐述,并就java8带来的3个注解做专题说明和举例:

@Repeatable注解,关于类型注解的声明,函数式接口注解@FunctionalInterface(与Lambdas结合使用)。

@Repeatable:说明该注解标识的注解可以多次使用到同一个元素的声明上。

看一个使用的例子。首先我们创造一个能容纳重复的注解的容器:

1

2

3

4

5

6

7

8

9/**

* Container for the {@link CanBeRepeated} Annotation containing a list of values

*/

@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.TYPE_USE )

public @interface RepeatedValues

{

CanBeRepeated[] value();

}

接着,创建注解本身,然后标记@Repeatable

1

2

3

4

5

6

7@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.TYPE_USE )

@Repeatable( RepeatedValues.class )

public @interface CanBeRepeated

{

String value();

}

最后,我们可以这样重复地使用:

1

2

3

4

5

6

7@CanBeRepeated("the color is green" )

@CanBeRepeated("the color is red" )

@CanBeRepeated("the color is blue" )

public class RepeatableAnnotated

{

}

如果我们尝试去掉@Repeatable

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.TYPE_USE )

public @interface CannotBeRepeated

{

String value();

}

@CannotBeRepeated("info" )

/*

* if we try repeat the annotation we will get an error: Duplicate annotation of non-repeatable type

*

* @CannotBeRepeated. Only annotation types marked

*

* @Repeatable can be used multiple times at one target.

*/

// @CannotBeRepeated( "more info" )

public class RepeatableAnnotatedWrong

{

}

我们会得到编译器的错误信息:

1Duplicate annotation of non-repeatable type

自Java8开始,我们可以在类型上使用注解。由于我们在任何地方都可以使用类型,包括 new操作符,casting,implements,throw等等。注解可以改善对Java代码的分析并且保证更加健壮的类型检查。这个例子说明了这一点:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19@SuppressWarnings("unused" )

public static void main(String[] args )

{

// type def

@TypeAnnotated

String cannotBeEmpty =null;

// type

List myList =new ArrayList();

// values

String myString =new @TypeAnnotatedString("this is annotated in java 8" );

}

// in method params

public void methodAnnotated( @TypeAnnotatedint parameter )

{

System.out.println("do nothing" );

}

所有的这些在Java8之前都是不可能的。

@FunctionalInterface:这个注解表示一个函数式接口元素。函数式接口是一种只有一个抽象方法(非默认)的接口。编译器会检查被注解元素,如果不符,就会产生错误。例子如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29// implementing its methods

@SuppressWarnings("unused" )

MyCustomInterface myFuncInterface =new MyCustomInterface()

{

@Override

public int doSomething(int param )

{

return param *10;

}

};

// using lambdas

@SuppressWarnings("unused" )

MyCustomInterface myFuncInterfaceLambdas = ( x ) -> ( x *10 );

}

@FunctionalInterface

interface MyCustomInterface

{

/*

* more abstract methods will cause the interface not to be a valid functional interface and

* the compiler will thrown an error:Invalid '@FunctionalInterface' annotation;

* FunctionalInterfaceAnnotation.MyCustomInterface is not a functional interface

*/

// boolean isFunctionalInterface();

int doSomething(int param );

}

这个注解可以被使用到类,接口,枚举和注解本身。它的被JVM保留并在runtime可见,这个是它的声明:

1

2

3

4@Documented

@Retention(value=RUNTIME)

@Target(value=TYPE)

public @interface FunctionalInterface

9. 自定义注解

正如我们之前多次提及的,可以定义和实现自定义注解。本章我们即将探讨。

首先,定义一个注解:

1public @interface CustomAnnotationClass

这样创建了一个新的注解类型名为 CustomAnnotationClass。关键字:@interface说明这是一个自定义注解的定义。

之后,你需要为此注解定义一对强制性的属性,保留策略和目标。还有一些其他属性可以定义,不过这两个是最基本和重要的。它们在第8章,描述注解的注解时讨论过,它们同样也是Java内建的注解。

所以,我们为自定义的注解设置属性:

1

2

3@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.TYPE )

public @interface CustomAnnotationClassimplements CustomAnnotationMethod

在保留策略中 RUNTIME 告诉编译器这个注解应该被被JVM保留,并且能通过反射在运行时分析。通过 TYPE 我们又设置该注解可以被使用到任何类的元素上。

之后,我们定义两个注解的成员:

1

2

3

4

5

6

7

8

9

10@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.TYPE )

public @interface CustomAnnotationClass

{

public String author()default "danibuiza";

public String date();

}

以上我们仅定义了默认值为“danibuiza”的 author 属性和没有默认值的date属性。我们应强调所有的方法声明都不能有参数和throw子句。这个返回值的类型被限制为之前提过的字符串,类,枚举,注解和存储这些类型的数组。

现在我们可以像这样使用刚创建的自定义注解:

1

2

3

4

5@CustomAnnotationClass( date ="2014-05-05" )

public class AnnotatedClass

{

...

}

在另一种类似的用法中我们可以创建一种注解方法的注解,使用Target METHOD:

1

2

3

4

5

6

7

8

9

10

11

12@Retention( RetentionPolicy.RUNTIME )

@Target( ElementType.METHOD )

public @interface CustomAnnotationMethod

{

public String author()default "danibuiza";

public String date();

public String description();

}

这种注解可以使用在方法声明上:

1

2

3

4

5

6

7

8

9

10

11@CustomAnnotationMethod( date ="2014-06-05", description ="annotated method" )

public String annotatedMethod()

{

return "nothing niente";

}

@CustomAnnotationMethod( author ="friend of mine", date ="2014-06-05", description ="annotated method" )

public String annotatedMethodFromAFriend()

{

return "nothing niente";

}

有很多其它属性可以用在自定义注解上,但是 目标 (Target)和 保留策略(Retention Policy)是最重要的两个。

10. 获取注解

Java反射API包含了许多方法来在运行时从类,方法或者其它元素获取注解。接口AnnotatedElement包含了大部分重要的方法,如下:

getAnnotations(): 返回该元素的所有注解,包括没有显式定义该元素上的注解。

isAnnotationPresent(annotation): 检查传入的注解是否存在于当前元素。

getAnnotation(class): 按照传入的参数获取指定类型的注解。返回null说明当前元素不带有此注解。

class 通过java.lang.Class被实现,java.lang.reflect.Method 和 java.lang.reflect.Field,所以可以基本上被和任何Java元素使用。

现在,我们将看一个怎么读取注解的例子:

我们写一个程序,从一个类和它的方法中读取所有的存在的注解:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36public static void main(String[] args ) throws Exception

{

Class object = AnnotatedClass.class;

// Retrieve all annotations from the class

Annotation[] annotations = object.getAnnotations();

for( Annotation annotation : annotations )

{

System.out.println( annotation );

}

// Checks if an annotation is present

if( object.isAnnotationPresent( CustomAnnotationClass.class ) )

{

// Gets the desired annotation

Annotation annotation = object.getAnnotation( CustomAnnotationClass.class );

System.out.println( annotation );

}

// the same for all methods of the class

for( Method method : object.getDeclaredMethods() )

{

if( method.isAnnotationPresent( CustomAnnotationMethod.class ) )

{

Annotation annotation = method.getAnnotation( CustomAnnotationMethod.class );

System.out.println( annotation );

}

}

}

输出如下:

1

2

3

4

5

6@com.danibuiza.javacodegeeks.customannotations.CustomAnnotationClass(getInfo=Info, author=danibuiza, date=2014-05-05)

@com.danibuiza.javacodegeeks.customannotations.CustomAnnotationClass(getInfo=Info, author=danibuiza, date=2014-05-05)

@com.danibuiza.javacodegeeks.customannotations.CustomAnnotationMethod(author=friend of mine, date=2014-06-05, description=annotated method)

@com.danibuiza.javacodegeeks.customannotations.CustomAnnotationMethod(author=danibuiza, date=2014-06-05, description=annotated method)

在这个程序中,我们可以看到 getAnnotations()方法来获取所有某个对象(方法,类)上的所有注解的用法。展示了怎样使用isAnnotationPresent()方法和getAnnotation()方法检查是否存在特定的注解,和如何获取它。

11. 注解中的继承

注解在Java中可以使用继承。这种继承和普通的面向对象继承几乎没有共同点。

如果一个注解在Java中被标识成继承,使用了保留注解@Inherited,说明它注解的这个类将自动地把这个注解传递到所有子类中而不用在子类中声明。通常,一个类继承了父类,并不继承父类的注解。这完全和使用注解的目的一致的:提供关于被注解的代码的信息而不修改它们的行为。

我们通过一个例子更清楚地说明。首先,我们定义一个自动继承的自定义注解。

1

2

3

4

5

6

7@Inherited

@Retention(RetentionPolicy.RUNTIME)

@Target(ElementType.TYPE)

public @interface InheritedAnnotation

{

}

有一个父类名为:AnnotatedSuperClass,已经被自定义的注解给注解上了:

1

2

3

4

5

6

7

8

9

10@InheritedAnnotation

public class AnnotatedSuperClass

{

public void oneMethod()

{

}

}

一个子类继承父类:

1

2

3

4

5

6

7

8

9

10@InheritedAnnotation

public class AnnotatedSuperClass

{

public void oneMethod()

{

}

}

子类 AnnotatedSubClass 展示了自动继承的注解 @InheritedAnnotation。我们看到下面的代码通过 isAnnotationPresent() 方法测试出了当前注解。

1

2

3

4

System.out.println("is true: " + AnnotatedSuperClass.class.isAnnotationPresent( InheritedAnnotation.class ) );

System.out.println("is true: " + AnnotatedSubClass.class.isAnnotationPresent( InheritedAnnotation.class ) );

输出如下:

1

2istrue:true

istrue:true

我们可以看到子类虽然并没有声明注解,但还是被自动地注解上了。

如果我们尝试注解在一个接口中:

1

2

3

4

5

6

7@InheritedAnnotation

public interface AnnotatedInterface

{

public void oneMethod();

}

一个实现了该接口的类:

1

2

3

4

5

6

7

8

9

10public class AnnotatedImplementedClassimplements AnnotatedInterface

{

@Override

public void oneMethod()

{

}

}

经过 isAnnotationPresent() 方法测试:

1

2

3System.out.println("is true: " + AnnotatedInterface.class.isAnnotationPresent( InheritedAnnotation.class ) );

System.out.println("is true: " + AnnotatedImplementedClass.class.isAnnotationPresent( InheritedAnnotation.class ) );

结果如下:

1

2is true:true

is true:false

这个结果说明继承注解和接口在一起使用时,接口中的注解在实现类中:仅仅被忽略。实现类并不继承接口的注解;接口继承仅仅适用于类继承。正如 AnnotatedSubClass。

@Inheriated注解仅在存在继承关系的类上产生效果,在接口和实现类上并不工作。这条同样也适用在方法,变量,包等等。只有类才和这个注解连用。

注解不能继承注解,如果你尝试这么做了,就会得到编译器抛出的错误:

1Annotation type declaration cannot have explicit superinterfaces

12. 使用注解的知名类库

在这一章我们将展示知名类库是如何利用注解的。一些类库如:JAXB, Spring Framework, Findbugs, Log4j, Hibernate, Junit。它们使用注解来完成代码质量分析,单元测试,XML解析,依赖注入和许多其它的工作。

在这篇手册中我们将讨论以下类库的部分内容:

12.1. Junit

这个框架用于完成Java中的单元测试。自JUnit4开始,注解被广泛应用,成为Junit的设计的主干之一。

基本上,JUnit处理程序通过反射读取类和测试套件,按照在方法上,类上的注解顺序地执行它们。当然还有一些用来修改测试执行的注解。其它注解都用来执行测试,阻止执行,改变执行顺序等等。

用到的注解相当多,但是我们将会看到最重要的几个:

@Test:这个注解向JUnit说明这个被注解的方法一定是一个可执行的测试方法。这个注解只能标识在方法上,并且被JVM保留至运行时。

1

2

3

4

5

6@Test

public void testMe()

{

//test assertions

assertEquals(1,1);

}

从这个例子可以看到我们如何在JUnit中使用这类注解。

@Before:这个注解用来向JUnit说明被标记的方法应该在所有测试方法之前被执行。这对于在测试之前设置测试环境和初始化非常有用。同样只适用于方法上:

1

2

3

4

5

6

7@Before

public void setUp()

{

// initializing variables

count =0;

init();

}

@After:这个注解用来向JUnit说明被注解的方法应该在所有单元测试之后执行。这个注解通常用来销毁资源,关闭,释放资源或者清理,重置等工作。

1

2

3

4

5

6@After

public void destroy()

{

// closing input stream

stream.close();

}

@Ignore:这个方法用来向JUnit说明被注解的方法应该不被当作测试单元执行。即使它被注解成为一个测试方法,也只能被忽略。

1

2

3

4

5

6

7@Ignore

@Test

public void donotTestMe()

{

count = -22;

System.out.println("donotTestMe(): " + count );

}

这个方法可能在在开发调试阶段使用,但一旦开始进入发布阶段变需要将被忽略的代码去掉。

@FixMethodOrder:指定执行的顺序,正常情况下Junit处理程序负责它按照完全随机的无法预知的顺序执行。当所有的测试方法都相互独立的时候,不推荐使用这个注解。但是,当测试的场景需要测试方法按照一定规则的时候,这个注解就派上用场了。

还有一些测试套件和类库使用了注解,例如Mockito和JMock,使用注解来创建测试对象和预期值。 查看JUnit中所有注解请参考:https://github.com/junit-team/junit/wiki/Getting-started

12.2. Hibernate ORM

Hibernate可能是一个用得最广泛的对象关系映射类库。它提供了对象模型和关系型数据库的映射框架。使用注解作为设计的一部分。

在本章我们将讨论一两个由Hibernate提供的注解并解释它的处理程序如何处理它们。

下面的代码段使用了@Entity和@Table。这两个是用来向消费器(Hibernate处理程序)说明被注解的类是一个实体类,以及它映射的SQL表名。实际上,这个注解仅仅是指明了主表,还可以有说明字表的注解。

1

2

3@Entity

@Table( name ="hibernate_annotated" )

public class HibernateAnnotated

接下来的代码段展示了如何向Hibernate处理程序说明被标记的元素是表的主键,并映射名为“id”的列,并且主键是自动生成的。

1

2

3

4@Id

@GeneratedValue

@Column( name ="id" )

private int id;

为了指定标准的SQL表列名,我们可以写如下注解:

1

2@Column( name ="description" )

private String description;

这说明被标记的元素映射的是这个类所对应的表中名为“description”的一列。

12.3. Spring MVC

Spring是个被广泛使用的Java企业级应用框架。其一项重要的特性就是在Java程序使用依赖注入。

Spring使用注解作为XML(Spring的早期版本使用的基于XML配置)的一种替代方式。现在两者都是可行的。你可以使用XML文件或者注解配置你的项目。在我看来两者都各有优势。

我们将在下例中展示两个可用注解:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18@Component

public class DependencyInjectionAnnotation

{

private String description;

public String getDescription()

{

return description;

}

@Autowired

public void setDescription(String description )

{

this.description = description;

}

}

在次代码片段中我们可以找到两个使用在了类和方法上的注解。

@Component:说明被标记的元素,在本例中是一个类,是一个自动检测的目标。这意味着被注解的类,将会被Spring容器实例化并管理。

@Autowired:Spring容器将会尝试通过类型(这是一种元素匹配机制)使用这个set方法来自动装配。此注解也可以使用在构造器和属性上,Spring也会根据注解的地方不同采取不同的操作。

12.4. Findbugs

这是一个用来测量代码质量,并提供一系列可能提高它的工具。它会根据预定义(或者自定义)的违反规则来检查代码。Findbugs提供一系列注解来允许开发者来改变默认行为。

它主要使用反射读取代码(和包含的注解)并决定基于它们,应该采取什么行为。

一个列子是 edu.umd.cs.findbugs.annotations.SuppressFBWarnings 注解期待一种违反规定的键值并把它当作参数。这非常像 java.lang.SuppressWarnings。被用来向 Findbugs 说明当执行代码分析的时候,忽略指定的违反规则。

这是一个例子:

1

2

3

4

5

6

7

8

9

10

11@SuppressFBWarnings("HE_EQUALS_USE_HASHCODE" )

public class FindBugsAnnotated

{

@Override

public boolean equals( Object arg0 )

{

return super.equals( arg0 );

}

}

这个类已经重现了 Object 的 equals 方法,但是并没有重写 hashCode 方法。这通常会导致问题,因为 hashCode 和 equals 两者应该被同时重写,否则在使用这个对象作为 HashMap 的key值的时候会导致错误。所以,Findbugs会在违反规则报告中创造一个错误条目。

如果注解 @SuppressFBWarnings 设置了 HE_EQUALS_USE_HASHCODE 值以后,处理程序就不会在抛出这类型的错误了。

1

2Bug: com.danibuiza.javacodegeeks.findbugsannotations.FindBugsAnnotated defines equals and usesObject.hashCode()

Bug: com.danibuiza.javacodegeeks.findbugsannotations.FindBugsAnnotated defines equals and usesObject.hashCode()

这个类重写了 equals 方法,但是没有重写 hashCode 方法,继承自 java.lang.Object 的 hashCode 方法(返回每个对象被JVM赋予的特定值)。因此,这个类几乎违反了相同的对象必须 hashcode 相同的一致性。

如果你认为这个类的实例不会插入到哈希表,推荐的做法是像这样实现 hashCode 方法:

1

2

3

4

5

6

7

8public int hashCode() {

assertfalse :"hashCode not designed";

return 42;// any arbitrary constant will do

}

Rank: Troubling (14), confidence: High

Pattern: HE_EQUALS_USE_HASHCODE

Type: HE, Category: BAD_PRACTICE (Bad practice)

这个错误包含了该问题的一种解释并且提示如何处理它。在这种情况下,解决方案应该是实现 hashCode 方法。

12.5. JAXB

JAXB是一个用来相互转换和映射XML文件与Java对象的类库。实际上,这个类库与标准JRE一起提供,不需要任何额外的下载和配置。可以直接通过引入 java.xml.bind.annotation 包下的类直接使用。

JAXB使用注解来告知处理程序(或者是JVM)XML文件与代码的相互转化。例如,注解可以用来在代码上标记XML节点,XMl属性,值等等。我们将看到一个例子:

首先,我们声明一个类说明它应该是XML文件中的一个节点:

1

2

3

4

5

6import javax.xml.bind.annotation.XmlRootElement;

import javax.xml.bind.annotation.XmlType;

@XmlType( propOrder = {"brand","model","year","km" } )

@XmlRootElement( name ="Car" )

class Car

...

使用注解@XmlType,@XmlRoootElement。它们用来告知 JAXB 处理程序 Car 这个类在转换后,将会转换成为XML中的一个节点。这个@XmlType说明了属性在XML中的顺序。JAXB将会基于这些注解执行合适的操作。

除了分离的 getter 和 setter 属性,再也不需要向这个类中添加其它东西来完成转换。现在我们需要一个消费器程序来执行转换成XML:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22Car car =new Car();

car.setBrand("Mercedes" );

car.setModel("SLK" );

car.setYear(2011 );

car.setKm(15000 );

Car carVW =new Car();

carVW.setBrand("VW" );

carVW.setModel("Touran" );

carVW.setYear(2005 );

carVW.setKm(150000 );

/* init jaxb marshaler */

JAXBContext jaxbContext = JAXBContext.newInstance( Car.class );

Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

/* set this flag to true to format the output */

jaxbMarshaller.setProperty( Marshaller.JAXB_FORMATTED_OUTPUT, true );

/* marshaling of java objects in xml (output to standard output) */

jaxbMarshaller.marshal( car, System.out );

jaxbMarshaller.marshal( carVW, System.out );

程序产生的输入如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15<?xml version="1.0" encoding="UTF-8" standalone="yes"?>

02

Mercedes

SLK

2011

15000

VW

Touran

2005

150000

更多关于JAXB XML和Java 的转换信息请参考:https://jaxb.java.net/

13. 总结

在这篇文档中,我们解释了Java注解是Java1.5开始一个非常重要的特性。基本上,注解都是作为包含代码信息的元数据而被标记到代码中。它们不会改变或者影响代码的任何意义,而是被第三方称为消费器的程序通过反射的方式使用。

我们列出了Java默认的内建注解,一些称为元注解例如:@Target或者 @Retention,又有@Override,@SuppressWarnings,还有一些Java8相关的注解,比如:@Repeatable,@FunctionalInterface和类型注解。我们还展现了一两个结合使用反射的例子,并描述了一些使用注解的类库例如Spring, Junit,Hibernate。

注解是Java中一种分析元数据的强大机制,可以在不同的程序中担任不同的作用,例如校验,依赖注入,单元测试。

14. 下载

这是一个java注解的教程。

15. 资料

这是一些非常有用的关于Java注解的资料:

java指导手册,Java 注解指导手册 – 终极向导相关推荐

  1. Java 注解指导手册 – 终极向导

    转载自  Java 注解指导手册 – 终极向导 译文出处: Toien Liu   原文出处:Dani Buiza 编者的话:注解是java的一个主要特性且每个java开发者都应该知道如何使用它. 我 ...

  2. java part part,java泛型指导手册(part1),指导手册part1

    java泛型指导手册(part1),指导手册part1 index 1.大致了解一下 从java5开始,泛型就被加入了java.泛型的意义在于:在你操作object之前,提供了一种指定特定类型为更加通 ...

  3. java判断闰年通过多态方法_04748《Java语言程序设计》实验指导书.doc

    04748<Java语言程序设计>实验指导书.doc Java语言程序设计 实验指导书 前言 一.上机实验目的 上机实验的目的是提高学生的分析问题.解决问题的能力和动手能力,通过实践环节理 ...

  4. python语言程序设计实践教程实验八答案_清华大学出版社-图书详情-《Java程序设计教程及实验指导》...

    前言 Java程序设计教程及实验指导Java是一门主流的面向对象的程序设计语言,是面向对象程序设计教学的主流语言.本书由有多年教学经验的专业教师编写,分为教程以及实验与课程设计两部分.教程部分从Jav ...

  5. 玄子Share-BCSP助学手册-JAVA开发

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-b2gPyAnt-1676810001349)(./assets/%E7%8E%84%E5%AD%90Share%E4%B ...

  6. 质量检验(三)——检验手册、检验指导书

    质量检验部门需要必备一份检验手册.如果你所在的质量检验部门还没有这样的指导性的文档,那就赶紧去编一本出来吧.因为,这是指导检验工作的方向和具体操作步骤. 今天就介绍一下检验手册和检验指导书       ...

  7. 年度第一首发。阿里Java顶级架构师面试标准手册对标P8,牛×啊

    年度第一首发!Java高级架构师面试技术标准手册(对标阿里P7架构师) 最新一年的金三银四面试跳槽季已经拉开序幕有些日子,很多小伙伴开始实行了跳槽工程.我找了很多资料把他总结了一套最新的面试题套餐资料 ...

  8. java程序设计题解与上机指导_Java程序设计习题解析与上机指导

    第一部分 习题与解析 第1章 Java语言概述 1.1 知识点回顾 1.2 典型题分析 结束语 结合练习一 一.选择题 二.填空题 三.判断题 四.程序设计题 第2章 Java语言基础 2.1 知识点 ...

  9. Java后台开发知识点快速复习手册导航页

    面试知识点复习手册 持续更新中,请查看专栏: [Java程序员面试助攻手册] 其中精选文章: [Python/Java]Leetcode题解分类汇总(前150题) Java基础知识点面试手册 快速梳理 ...

最新文章

  1. 利用Python实现十大经典排序算法(附代码流程)
  2. linux下使用syslog日志调试程序快速的调试代码信息的过程
  3. 记一次MongoDB性能问题(从MySQL迁移到MongoDB)
  4. Cisco 双出口NAT配置
  5. 申通快递机器人上岗_申通快速分拣机器人未来有70%人会失业
  6. 32位oracle_oracle 性能调优
  7. vb treeview 展开子节点_C# / VB.NET 在PPT中创建、编辑PPT SmartArt图形
  8. Namenode双机热备之Pacemaker
  9. 关于java.lang.NoClassDefFoundError的问题
  10. FindChildControl与FindComponent(动态创建的控件要通过Owner.FindComponent去找该控件)
  11. 指数体系与因素分析:概念、作用与种类
  12. SQUEEZENET: ALEXNET-LEVEL ACCURACY WITH 50X FEWER PARAMETERS AND 0.5MB MODEL SIZE
  13. halcon 深度学习标注_halcon深度学习: 分类
  14. Mortal Kombat Tower(dp)
  15. 【风马一族_php】PHP运算
  16. 圆形区域函数可视化和泊松方程求解
  17. 硬核,这 3 款 IDE 插件让你的代码牢不可破
  18. 羊毛地毯可以用水洗吗
  19. superset设置起止时间为明天
  20. 尚硅谷-宋红康-MySQL高级性能篇

热门文章

  1. 往有序链表的插入元素使原链表依旧有序
  2. 【数学与算法】凸函数、凸集、凸函数的一二阶数学解释
  3. 机器学习实战笔记(Python实现)-02-k近邻算法(kNN)
  4. 级联分类器训练及其参数的含义
  5. Vue.js 过滤器
  6. Hello World With JBoss Modules
  7. 深入理解MySQL的外连接、内连接、交叉连接
  8. Java面试参考指南——同步
  9. 彩色RGB图像转为灰度图像
  10. 分享一些优秀有趣的博客