注解,也被称为元数据,为咱们在代码中添加信息提供了一种形式化的方法,是咱们能够在稍后某个时刻很是方便地使用这些数据。php
注解在必定程度上是把元数据与源代码文件结合在一块儿,而不是保存在外部的文档中这一大的趋势之下所催生的。同时,注解也是来仔像C#之类的其余语言对Java形成的语言特性压力所作出的一种回应。html
注解是众多引入到Java SE5中的重要的语言变化之一。它们能够提供用来完整的描述程序所需的信息,而这些信息是没法用Java来表达的。所以,注解是得咱们可以以将由编译器来测试和验证的格式、存储有关程序的额外信息。java
---《Java编程思想-第4版》程序员
注解的语法比较简单,以“@”符号开始,Java SE5内置提供了三种,定义在java.lang中的注解:编程
Java还另外提供了四种注解,专门负责新注解的建立。注解的定义看起来很像接口发的定义,与其余任何Java接口同样,注解也将会编译成class文件:框架
package com.phpdragon.samples; import java.lang.annotation; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface Test {}
除了@符号外,@Test的定义很像一个空的接口。定义注解时,须要一些元注解(meta-annotation),如@Target和@Retention,@Target用来定义你的注解将用在什么地方,是方法前仍是域前,或者类前。ide
@Retention用来定义该注解在哪个级别可用,在源代码中(SOURCE),类文件中(CLASS)或者运行时(RUNTIME)。工具
四种元注解分别是@Target,@Retention,@Documented,@Inherited。性能
@Target注解 ,表示注解可用于什么地方。可能的ElementType参数包括:测试
@Retention注解 ,表示须要什么级别保存该注释信息。可选的RetentionPolicy参数包括:
@Documented注解
Documented 注解代表这个注解应该被 javadoc工具记录。默认状况下,javadoc是不包括注解的,但若是声明注解时指定了 @Documented,则它会被 javadoc 之类的工具处理,
因此注解类型信息也会被包括在生成的文档中,标注此类接口、方法、字段已经被废止。
@Inherited注解
容许子类继承父类中的注解。
使用注解的过程当中,很重要的一个部分就是建立与使用注解处理器。下面经过一个完整的例子来讲明怎么定义与使用注解处理器。
1.自定义注解类
import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface UseCase{ int id(); String description() default "no description"; }
2.使用自定义注解
public class PasswordUtils { @UseCase(id=1,description = "Passwords must contain at least one numeric") public boolean validatePassword(String password){ return (password.matches("\\w*\\d\\w*")); } @UseCase(id=2) public String encryptPassword(String password){ return new StringBuilder(password).reverse().toString(); } @UseCase(id=3,description = "New passwords can't equal previously used ones") public boolean checkForNewPassword(List<String> prevPasswords,String password){ return !prevPasswords.contains(password); } }
3.自定义注解处理器
import java.lang.reflect.Method; import java.util.ArrayList; import java.util.Collections; import java.util.List; public class UseCaseTracker { public static void trackUseCases(List<Integer> useCases,Class<?> cl){ for(Method m:cl.getDeclaredMethods()){ UseCase uc=m.getAnnotation(UseCase.class); if(uc != null){ System.out.println("Found Use Case:"+uc.id()+" "+uc.description()); useCases.remove(new Integer(uc.id())); } } for(int i:useCases){ System.out.println("Warning: Missing use case-"+i); } } public static void main(String[] args){ List<Integer> useCases=new ArrayList<Integer>(); Collections.addAll(useCases,1,2,3,4); trackUseCases(useCases,PasswordUtils.class); } }
4.结果打印输出
注解的元素在使用时表现为名-值对的形式,并须要置于@UseCase声明以后的括号内。在encryptPassword()方法的注解中,并无给出description元素的值,所以,在UseCase的注解处理器分析处理这个类时会使用该元素的默认值。
经过上面的例子能够看到,注解处理器用了两个反射方法getDeclaredMethods()和getAnnotation(),它们都属于AnnotatedElement接口(Class、Method与Field等类都实现了该接口)。
注解一般是跟反射机制一块儿使用的。
利用Java SE5提供的注解和反射机制,就能实现你们熟知的 DI(依赖注入) 和 AOP(面向切面编程),大名鼎鼎Spring框架就是注解使用上炉火纯青的好例子。
抱歉,就目前出现的PHP五、7版本而言,还未提供上述Java的内置元注解和注解概念。
但能够经过PHP的ReflectionClass类来解析PHP的代码注释,从而实现一套本身的注解机制。如这位同窗的例子:PHP反射机制实现自动依赖注入。
但并不建议使用反射来实现注解机制,从而实现 IOC 和 AOP功能。
一个缘由是PHP对注释并未提供语法检查机制,没法形式化,须要开发框架和程序员来约定注释。
二是Java中的AOP,在字节码层面上有代码织入等技术支持,从而提升了程序性能。
一般的作法是在框架层面进行hook处理,举例连接中使用的就是这个大体方式。
PS: