註解,也被稱爲元數據,爲咱們在代碼中添加信息提供了一種形式化的方法,是咱們能夠在稍後某個時刻很是方便地使用這些數據。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: