這篇文章主要介紹“如何使用自定義注解”,在日常操作中,相信很多人在如何使用自定義注解問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”如何使用自定義注解”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
盤州網(wǎng)站制作公司哪家好,找成都創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、成都響應(yīng)式網(wǎng)站建設(shè)公司等網(wǎng)站項目制作,到程序開發(fā),運營維護。成都創(chuàng)新互聯(lián)成立于2013年到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運維經(jīng)驗,來保證我們的工作的順利進行。專注于網(wǎng)站建設(shè)就選成都創(chuàng)新互聯(lián)。
①、引用自維基百科的內(nèi)容:
Java注解又稱Java標(biāo)注,是JDK5.0版本開始支持加入源代碼的特殊語法 元數(shù)據(jù)。
Java語言中的類、方法、變量、參數(shù)和包等都可以被標(biāo)注。和Javadoc不同,Java標(biāo)注可以通過反射獲取標(biāo)注內(nèi)容。在編譯器生成類文件時,標(biāo)注可以被嵌入到字節(jié)碼中。Java虛擬機可以保留標(biāo)注內(nèi)容,在運行時可以獲取到標(biāo)注內(nèi)容。 當(dāng)然它也支持自定義Java標(biāo)注。
②、引用自網(wǎng)絡(luò)的內(nèi)容:
Java 注解是在 JDK5 時引入的新特性,注解(也被稱為 元數(shù)據(jù))為我們在代碼中添加信息提供了一種形式化的方法,使我們可以在稍后某個時刻非常方便地使用這些數(shù)據(jù)。
元注解的作用就是負責(zé)注解其他注解。Java5.0定義了4個標(biāo)準(zhǔn)的meta-annotation(元注解)類型,它們被用來提供對其它 annotation類型作說明。
標(biāo)準(zhǔn)的元注解:
@Target
@Retention
@Documented
@Inherited
在詳細說這四個元數(shù)據(jù)的含義之前,先來看一個在工作中會經(jīng)常使用到的 @Autowired 注解,進入這個注解里面瞧瞧: 此注解中使用到了@Target、@Retention、@Documented 這三個元注解 。
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface Autowired { boolean required() default true; }
@Target元注解:
@Target注解,是專門用來限定某個自定義注解能夠被應(yīng)用在哪些Java元素上面的,標(biāo)明作用范圍;取值在
java.lang.annotation.ElementType
進行定義的。
public enum ElementType { /** 類,接口(包括注解類型)或枚舉的聲明 */ TYPE, /** 屬性的聲明 */ FIELD, /** 方法的聲明 */ METHOD, /** 方法形式參數(shù)聲明 */ PARAMETER, /** 構(gòu)造方法的聲明 */ CONSTRUCTOR, /** 局部變量聲明 */ LOCAL_VARIABLE, /** 注解類型聲明 */ ANNOTATION_TYPE, /** 包的聲明 */ PACKAGE }
根據(jù)此處可以知道 @Autowired 注解的作用范圍:
// 可以作用在 構(gòu)造方法、方法、方法形參、屬性、注解類型 上@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})
@Retention元注解:
@Retention注解,翻譯為持久力、保持力。即用來修飾自定義注解的生命周期。
注解的生命周期有三個階段:
Java源文件階段;
編譯到class文件階段;
運行期階段;
同樣使用了RetentionPolicy 枚舉類型對這三個階段進行了定義:
public enum RetentionPolicy { /** * Annotations are to be discarded by the compiler. * (注解將被編譯器忽略掉) */ SOURCE, /** * Annotations are to be recorded in the class file by the compiler * but need not be retained by the VM at run time. This is the default * behavior. * (注解將被編譯器記錄在class文件中,但在運行時不會被虛擬機保留,這是一個默認的行為) */ CLASS, /** * Annotations are to be recorded in the class file by the compiler and * retained by the VM at run time, so they may be read reflectively. * (注解將被編譯器記錄在class文件中,而且在運行時會被虛擬機保留,因此它們能通過反射被讀取到) * @see java.lang.reflect.AnnotatedElement */ RUNTIME }
再詳細描述下這三個階段:
①、如果被定義為 RetentionPolicy.SOURCE,則它將被限定在Java源文件中,那么這個注解即不會參與編譯也不會在運行期起任何作用,這個注解就和一個注釋是一樣的效果,只能被閱讀Java文件的人看到;
②、如果被定義為 RetentionPolicy.CLASS,則它將被編譯到Class文件中,那么編譯器可以在編譯時根據(jù)注解做一些處理動作,但是運行時JVM(Java虛擬機)會忽略它,并且在運行期也不能讀取到;
③、如果被定義為 RetentionPolicy.RUNTIME,那么這個注解可以在運行期的加載階段被加載到Class對象中。那么在程序運行階段,可以通過反射得到這個注解,并通過判斷是否有這個注解或這個注解中屬性的值,從而執(zhí)行不同的程序代碼段。
注意:實際開發(fā)中的自定義注解幾乎都是使用的 RetentionPolicy.RUNTIME。
@Documented元注解:
@Documented注解,是被用來指定自定義注解是否能隨著被定義的java文件生成到JavaDoc文檔當(dāng)中。
@Inherited元注解:
@Inherited注解,是指定某個自定義注解如果寫在了父類的聲明部分,那么子類的聲明部分也能自動擁有該注解。
@Inherited注解只對那些@Target被定義為 ElementType.TYPE 的自定義注解起作用。
在了解了上面的內(nèi)容后,我們來嘗試實現(xiàn)一個自定義注解:
根據(jù)上面自定義注解中使用到的元注解得知:
①、此注解的作用范圍,可以使用在類(接口、枚舉)、方法上;
②、此注解的生命周期,被編譯器保存在class文件中,而且在運行時會被JVM保留,可以通過反射讀?。?/p>
上面已經(jīng)創(chuàng)建了一個自定義的注解,那該怎么使用呢?下面首先描述下它簡單的用法,后面將會使用其結(jié)合攔截器和AOP切面編程進行實戰(zhàn)應(yīng)用;
使用自定義注解 結(jié)合 攔截器 優(yōu)雅的實現(xiàn)對API接口響應(yīng)的包裝。
在介紹自定義實現(xiàn)的方式之前,先簡單介紹下普遍的實現(xiàn)方式,通過兩者的對比,才能更加明顯的發(fā)現(xiàn)誰最優(yōu)雅。
普通的接口響應(yīng)包裝方式:
現(xiàn)在項目絕大部分都采用的前后端分離方式,所以需要前端和后端通過接口進行交互;目前在接口交互中使用最多的數(shù)據(jù)格式是 json,然后后端返回給前端的最為常見的響應(yīng)格式如下:
{ #返回狀態(tài)碼 code:integer, #返回信息描述 message:string, #返回數(shù)據(jù)值 data:object }
項目中經(jīng)常使用枚舉類定義狀態(tài)碼和消息,代碼如下:
/** * @author 【 木子雷 】 公眾號 * @Title: ResponseCode * @Description: 使用枚舉類封裝好的響應(yīng)狀態(tài)碼及對應(yīng)的響應(yīng)消息 * @date: 2019年8月23日 下午7:12:50 */public enum ResponseCode { SUCCESS(1200, "請求成功"), ERROR(1400, "請求失敗"); private Integer code; private String message; private ResponseCode(Integer code, String message) { this.code = code; this.message = message; } public Integer code() { return this.code; } public String message() { return this.message; } }
同時項目中也會設(shè)計一個返回響應(yīng)包裝類,代碼如下:
import com.alibaba.fastjson.JSONObject;import java.io.Serializable;/** * @author 【 木子雷 】 公眾號 * @Title: Response * @Description: 封裝的統(tǒng)一的響應(yīng)返回類 * @date: 2019年8月23日 下午7:07:13 */@SuppressWarnings("serial")public class Responseimplements Serializable { /** * 響應(yīng)數(shù)據(jù) */ private T date; /** * 響應(yīng)狀態(tài)碼 */ private Integer code; /** * 響應(yīng)描述信息 */ private String message; public Response(T date, Integer code, String message) { super(); this.date = date; this.code = code; this.message = message; } public T getDate() { return date; } public void setDate(T date) { this.date = date; } public Integer getCode() { return code; } public void setCode(Integer code) { this.code = code; } public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } @Override public String toString() { return JSONObject.toJSONString(this); } }
最后就是使用響應(yīng)包裝類和狀態(tài)碼枚舉類 來實現(xiàn)返回響應(yīng)的包裝了:
@GetMapping("/user/findAllUser")public Response> findAllUser() { logger.info("開始查詢所有數(shù)據(jù)..."); List
findAllUser = new ArrayList<>(); findAllUser.add(new User("木子雷", 26)); findAllUser.add(new User("公眾號", 28)); // 返回響應(yīng)進行包裝 Response response = new Response(findAllUser, ResponseCode.SUCCESS.code(), ResponseCode.SUCCESS.message()); logger.info("response: {} \n", response.toString()); return response; }
在瀏覽器中輸入網(wǎng)址: http://127.0.0.1:8080/v1/api/user/findAllUser 然后點擊回車,得到如下數(shù)據(jù):
{ "code": 1200, "date": [ { "age": 26, "name": "木子雷" }, { "age": 28, "name": "公眾號" } ], "message": "請求成功"}
通過看這中實現(xiàn)響應(yīng)包裝的方式,我們能發(fā)現(xiàn)什么問題嗎?
答:代碼很冗余,需要在每個接口方法中都進行響應(yīng)的包裝;使得接口方法包含了很多非業(yè)務(wù)邏輯代碼;
有沒有版本進行優(yōu)化下呢? en en 思考中。。。。。 啊,自定義注解 + 攔截器可以實現(xiàn)呀!
自定義注解實現(xiàn)接口響應(yīng)包裝:
①、首先創(chuàng)建一個進行響應(yīng)包裝的自定義注解:
/** * @author 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.annotation * @ClassName: ResponseResult * @Description: 標(biāo)記方法返回值需要進行包裝的 自定義注解 * @Date: 2020-11-10 10:38 **/@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface ResponseResult { }
②、創(chuàng)建一個攔截器,實現(xiàn)對請求的攔截,看看請求的方法或類上是否使用了自定義的注解:
/** * @author 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.interceptor * @ClassName: ResponseResultInterceptor * @Description: 攔截器:攔截請求,判斷請求的方法或類上是否使用了自定義的@ResponseResult注解, * 并在請求內(nèi)設(shè)置是否使用了自定義注解的標(biāo)志位屬性; * @Date: 2020-11-10 10:50 **/@Componentpublic class ResponseResultInterceptor implements HandlerInterceptor { /** * 標(biāo)記位,標(biāo)記請求的controller類或方法上使用了到了自定義注解,返回數(shù)據(jù)需要被包裝 */ public static final String RESPONSE_ANNOTATION = "RESPONSE_ANNOTATION"; /** * 請求預(yù)處理,判斷是否使用了自定義注解 */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 請求的接口方法 if (handler instanceof HandlerMethod) { final HandlerMethod handlerMethod = (HandlerMethod) handler; final Class> clazz = handlerMethod.getBeanType(); final Method method = handlerMethod.getMethod(); // 判斷是否在類對象上加了注解 if (clazz.isAnnotationPresent(ResponseResult.class)) { // 在請求中設(shè)置需要進行響應(yīng)包裝的屬性標(biāo)志,在下面的ResponseBodyAdvice增強中進行處理 request.setAttribute(RESPONSE_ANNOTATION, clazz.getAnnotation(ResponseResult.class)); } else if (method.isAnnotationPresent(ResponseResult.class)) { // 在請求中設(shè)置需要進行響應(yīng)包裝的屬性標(biāo)志,在下面的ResponseBodyAdvice增強中進行處理 request.setAttribute(RESPONSE_ANNOTATION, method.getAnnotation(ResponseResult.class)); } } return true; } }
③、創(chuàng)建一個增強Controller,實現(xiàn)對返回響應(yīng)進行包裝的增強處理:
/** * @author 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.interceptor * @ClassName: ResponseResultHandler * @Description: 對 返回響應(yīng) 進行包裝 的增強處理 * @Date: 2020-11-10 13:49 **/@ControllerAdvicepublic class ResponseResultHandler implements ResponseBodyAdvice
④、最后在 Controller 中使用上我們的自定義注解;在 Controller 類上或者 方法上使用@ResponseResult自定義注解即可; 在瀏覽器中輸入網(wǎng)址: http://127.0.0.1:8080/v1/api/user/findAllUserByAnnotation 進行查看:
// 自定義注解用在了方法上@ResponseResult@GetMapping("/user/findAllUserByAnnotation")public ListfindAllUserByAnnotation() { logger.info("開始查詢所有數(shù)據(jù)..."); List findAllUser = new ArrayList<>(); findAllUser.add(new User("木子雷", 26)); findAllUser.add(new User("公眾號", 28)); logger.info("使用 @ResponseResult 自定義注解進行響應(yīng)的包裝,使controller代碼更加簡介"); return findAllUser; }
至此我們的接口返回響應(yīng)包裝自定義注解實現(xiàn)設(shè)計完成,看看代碼是不是又簡潔,又優(yōu)雅呢。
總結(jié):本文針對此方案只是進行了簡單的實現(xiàn),如果有興趣的朋友可以進行更好的優(yōu)化。
分布式鎖的最常見的使用流程:
通過上面的代碼可以得到一個信息:如果有很多方法中需要使用分布式鎖,那么每個方法中都必須有獲取分布式鎖和釋放分布式鎖的代碼,這樣一來就會出現(xiàn)代碼冗余;
那有什么好的解決方案嗎? 自定義注解使代碼變得更加簡潔、優(yōu)雅;
自定義注解優(yōu)雅的使用分布式鎖:
①、首先實現(xiàn)一個標(biāo)記分布式鎖使用的自定義注解:
/** * @author 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.annotation * @ClassName: GetDistributedLock * @Description: 獲取redis分布式鎖 注解 * @Date: 2020-11-10 16:24 **/@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface GetDistributedLock { // 分布式鎖 key String lockKey(); // 分布式鎖 value,默認為 lockValue String lockValue() default "lockValue"; // 過期時間,默認為 300秒 int expireTime() default 300; }
②、定義一個切面,在切面中對使用了 @GetDistributedLock 自定義注解的方法進行環(huán)繞增強通知:
/** * @author: 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.aop * @ClassName: DistributedLockAspect * @Description: 自定義注解結(jié)合AOP切面編程優(yōu)雅的使用分布式鎖 * @Date: 2020-11-10 16:52 **/@Component@Aspectpublic class DistributedLockAspect { private final Logger logger = LoggerFactory.getLogger(this.getClass()); @Autowired RedisService redisService; /** * Around 環(huán)繞增強通知 * * @param joinPoint 連接點,所有方法都屬于連接點;但是當(dāng)某些方法上使用了@GetDistributedLock自定義注解時, * 則其將連接點變?yōu)榱饲悬c;然后在切點上織入額外的增強處理;切點和其相應(yīng)的增強處理構(gòu)成了切面Aspect 。 */ @Around(value = "@annotation(com.lyl.annotation.GetDistributedLock)") public Boolean handlerDistributedLock(ProceedingJoinPoint joinPoint) { // 通過反射獲取自定義注解對象 GetDistributedLock getDistributedLock = ((MethodSignature) joinPoint.getSignature()) .getMethod().getAnnotation(GetDistributedLock.class); // 獲取自定義注解對象中的屬性值 String lockKey = getDistributedLock.lockKey(); String LockValue = getDistributedLock.lockValue(); int expireTime = getDistributedLock.expireTime(); if (redisService.tryGetDistributedLock(lockKey, LockValue, expireTime)) { // 獲取分布式鎖成功后,繼續(xù)執(zhí)行業(yè)務(wù)邏輯 try { return (boolean) joinPoint.proceed(); } catch (Throwable throwable) { logger.error("業(yè)務(wù)邏輯執(zhí)行失敗。", throwable); } finally { // 最終保證分布式鎖的釋放 redisService.releaseDistributedLock(lockKey, LockValue); } } return false; } }
③、最后,在 Controller 中的方法上使用 @GetDistributedLock 自定義注解即可;當(dāng)某個方法上使用了 自定義注解,那么這個方法就相當(dāng)于一個切點,那么就會對這個方法做環(huán)繞(方法執(zhí)行前和方法執(zhí)行后)增強處理;
在瀏覽器中輸入網(wǎng)址: http://127.0.0.1:8080/v1/api/user/getDistributedLock 回車后觸發(fā)方法執(zhí)行:
// 自定義注解的使用@GetDistributedLock(lockKey = "userLock")@GetMapping("/user/getDistributedLock")public boolean getUserDistributedLock() { logger.info("獲取分布式鎖..."); // 寫具體的業(yè)務(wù)邏輯 return true; }
通過自定義注解的方式,可以看到代碼變得更加簡潔、優(yōu)雅。
先看看最為常見的日志打印的方式,然后再聊聊自定義注解怎么優(yōu)雅的實現(xiàn)日志的打印。
普通日志的打印方式:
通過看上面的代碼可以知道,如果每個方法都需要打印下日志,那將會存在大量的冗余代碼;
自定義注解實現(xiàn)日志打?。?/strong>
①、首先創(chuàng)建一個標(biāo)記日志打印的自定義注解:
/** * @Author: 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.annotation * @ClassName: PrintLog * @Description: 自定義注解實現(xiàn)日志打印 * @Date: 2020-11-10 18:05 **/@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface PrintLog { }
②、定義一個切面,在切面中對使用了 @PrintLog 自定義注解的方法進行環(huán)繞增強通知:
/** * @author: 【 木子雷 】 公眾號 * @PACKAGE_NAME: com.lyl.aop * @ClassName: PrintLogAspect * @Description: 自定義注解結(jié)合AOP切面編程優(yōu)雅的實現(xiàn)日志打印 * @Date: 2020-11-10 18:11 **/@Component@Aspectpublic class PrintLogAspect { private final Logger logger = LoggerFactory.getLogger(this.getClass()); /** * Around 環(huán)繞增強通知 * * @param joinPoint 連接點,所有方法都屬于連接點;但是當(dāng)某些方法上使用了@PrintLog自定義注解時, * 則其將連接點變?yōu)榱饲悬c;然后在切點上織入額外的增強處理;切點和其相應(yīng)的增強處理構(gòu)成了切面Aspect 。 */ @Around(value = "@annotation(com.lyl.annotation.PrintLog)") public Object handlerPrintLog(ProceedingJoinPoint joinPoint) { // 獲取方法的名稱 String methodName = joinPoint.getSignature().getName(); // 獲取方法入?yún)?nbsp; Object[] param = joinPoint.getArgs(); StringBuilder sb = new StringBuilder(); for (Object o : param) { sb.append(o + "; "); } logger.info("進入《{}》方法, 參數(shù)為: {}", methodName, sb.toString()); Object object = null; // 繼續(xù)執(zhí)行方法 try { object = joinPoint.proceed(); } catch (Throwable throwable) { logger.error("打印日志處理error。。", throwable); } logger.info("{} 方法執(zhí)行結(jié)束。。", methodName); return object; } }
③、最后,在 Controller 中的方法上使用 @PrintLog 自定義注解即可;當(dāng)某個方法上使用了 自定義注解,那么這個方法就相當(dāng)于一個切點,那么就會對這個方法做環(huán)繞(方法執(zhí)行前和方法執(zhí)行后)增強處理;
@PrintLog@GetMapping(value = "/user/findUserNameById/{id}", produces = "application/json;charset=utf-8")public String findUserNameById(@PathVariable("id") int id) { // 模擬根據(jù)id查詢用戶名 String userName = "木子雷 公眾號"; return userName; }
④、在瀏覽器中輸入網(wǎng)址: http://127.0.0.1:8080/v1/api/user/findUserNameById/66 回車后觸發(fā)方法執(zhí)行,發(fā)現(xiàn)控制臺打印了日志:
進入《findUserNameById》方法, 參數(shù)為: 66; findUserNameById 方法執(zhí)行結(jié)束。。
到此,關(guān)于“如何使用自定義注解”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
本文標(biāo)題:如何使用自定義注解
鏈接地址:http://weahome.cn/article/pdppej.html