自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

注解都不會(huì)用,怎讓Java對你動(dòng)情

開發(fā) 后端
「注解」 也稱為 元數(shù)據(jù)。為我們在代碼中添加信息提供了一種形式化的方法,使我們可以在稍后某個(gè)時(shí)刻非常方便地使用這些數(shù)據(jù)。

[[344016]]

本文轉(zhuǎn)載自微信公眾號「小菜良記」,作者蔡不菜丶 。轉(zhuǎn)載本文請聯(lián)系小菜良記公眾號。   

初識(shí)

“「注解」 也稱為 元數(shù)據(jù)。為我們在代碼中添加信息提供了一種形式化的方法,使我們可以在稍后某個(gè)時(shí)刻非常方便地使用這些數(shù)據(jù)。

注解是在 Java 1.5 之后引入的,它可以提供用來完整地描述程序中所需的信息,可以由編譯器來測試和驗(yàn)證的格式,存儲(chǔ)有關(guān)程序的額外信息。

注解的使用很簡單,只需要和 @ 符號搭配。有些 Java 初學(xué)者常常會(huì)把 「注解」 和「注釋」 混淆,但是兩者的作用卻大同小異,都是用來描述信息,不同的是 「注解」 描述的信息是給應(yīng)用程序看的,而 「注釋」 描述的信息是給開發(fā)人員看的。

初學(xué)者對「注解」的印象可能不深,「注解」也許不起眼但是處處可見。

最常見到的便是 @Override,表示當(dāng)前的方法定義將覆蓋父類的方法,如果拼寫錯(cuò)誤,或者方法簽名對不上被覆蓋的方法,編譯器就會(huì)發(fā)出錯(cuò)誤提示。

既然都說到@Override注解了,那仔細(xì)回憶下,腦子里估計(jì)就浮現(xiàn)出@SuppressWarnnings注解。還記得我最早見到這個(gè)注解的時(shí)候,還是Myeclipse提示我使用的,我也不管三七二十一,就給標(biāo)注上了。后來才知道這個(gè)注解是用來關(guān)閉編譯器對類、方法、成員變量、變量初始化的警告。

上面兩個(gè)都說完了,那不妨再來一個(gè),這個(gè)可能是最不常見的。那就是@Deprecated

 

不要因?yàn)檫@個(gè)類加了橫線感到奇怪,那是因?yàn)锧Deprecated的效果。它的具體功能就是用于標(biāo)識(shí)方法或類,標(biāo)識(shí)該類或方法已過時(shí),建議不要使用。如果開發(fā)人員使用了注解為它的元素,那么編譯器就會(huì)發(fā)出警告信息。

剛剛開局不久,我們就已經(jīng)學(xué)到了三個(gè)注解的使用,雖然只是基本的,但接下來我們就通過這三個(gè)注解為導(dǎo)線通關(guān)打怪吧。

首關(guān)之程媧造注

注解一旦構(gòu)造出來,就享有編譯器的類型檢查保護(hù)。讓我們先看下一組代碼熱熱身:

  1. public class TestService { 
  2.  
  3.     @MyAnnotation 
  4.     public void runTset() { 
  5.         System.out.println("annotation test"); 
  6.     } 
  7.  

不要納悶,Java 中確實(shí)沒有一個(gè)注解名為MyAnnotation,但這個(gè)怎么來的呢,就是我們自己造的。

那么關(guān)子賣完了,接下來就來揭秘注解的制造:

  1. @Target(ElementType.METHOD) 
  2. @Retention(RetentionPolicy.RUNTIME) 
  3. public @interface MyAnnotation { 

這樣子,一個(gè)簡單的注解就新鮮出爐了。需要注意的是這可不是一個(gè)接口,需要留言interface前面還有一個(gè)@,這個(gè)標(biāo)識(shí)要是漏掉了,那可以天差地別。

細(xì)心的小伙伴可能注意到了,定義的注解頭上怎么還有注解。

 

這就是接下來要講到,敲黑板,注意看!

元注解來幫忙

在定義注解時(shí),會(huì)需要一些元注解。上面出現(xiàn)了兩個(gè),分別是@Target和@Retention.

其中@Target用來定義你的注解將應(yīng)用于什么地方(例如一個(gè)方法或一個(gè)域),@Retention用來定義該注解在哪一個(gè)級別可用,在源代碼中(「SOURCE」),類文件中(「CLASS」)或者運(yùn)行時(shí)(「RUNTIME」)。Java 提供了四種元注解,如下:

名稱 用處
「@Target」 標(biāo)識(shí)該注解可以用于什么地方。其中 ElementType 參數(shù)包括:
1. CONSTARUCTOR:構(gòu)造器的聲明
2. FIELD:域聲明(包括enum實(shí)例)
3. LOCAL_VARIABLE:局部變量聲明
4. METHOD:方法聲明
5. PACKAGE:包聲明
6. TYPE:類、接口(包括注解類型)或enum 聲明
「@Retention」 表示需要在什么級別保存該注解信息,其中RetentionPolicy參數(shù)包括:
1.SOURCE:注解將被編譯器丟棄
2.CLASS:注解在 class 文件中可用,但會(huì)被 VM 丟棄
3. RUNTIME:VM 將在運(yùn)行期也保留注解,因此可以通過反射機(jī)制讀取注解的信息
「@Documented」 將此注解包含在 JavaDoc 中
「@Inherited」 允許子類繼承父類的注解

注解也分類

我們在上面示例中創(chuàng)建了一個(gè) @MyAnnotation 注解??雌饋砗芎唵?,沒什么內(nèi)容,因此這種注解我們也稱為 「標(biāo)記注解」,注解也分幾類:

  • 標(biāo)記注解:注解內(nèi)部沒有屬性。使用方式:「@注解名」
  1. //定義 
  2. @Target(ElementType.METHOD) 
  3. @Retention(RetentionPolicy.RUNTIME) 
  4. public @interface MyAnnotation { 
  • 單值注解:注解內(nèi)部只有一個(gè)屬性。使用方式:「@注解名(key = value)」
  1. //定義 
  2. @Target(ElementType.METHOD) 
  3. @Retention(RetentionPolicy.RUNTIME) 
  4. public @interface SingleValueAnnotation { 
  5.     String name(); 
  6. //使用 
  7. @SingleValueAnnotation(name = "test"
  8. public void singleTest() {} 
  • 多值注解:注解內(nèi)部有過個(gè)屬性。使用方式:「@注解名(key = value, key = value, ...)」
  1. //定義 
  2. @Target(ElementType.METHOD) 
  3. @Retention(RetentionPolicy.RUNTIME) 
  4. public @interface MultiValueAnnotation { 
  5.     String name(); 
  6.     int num(); 
  7. //使用 
  8. @MultiValueAnnotation(name = "test", num = 1) 
  9. public void multiTest() {} 

值也有默認(rèn)

當(dāng)我們使用的不是標(biāo)記注解時(shí),如果在使用注解的時(shí)候不給注解中的屬性賦上值,那么編譯器就會(huì)報(bào)錯(cuò),提示我們需要賦值。

 

這樣子是很不方便,有時(shí)候我們并沒有使用到或值是固定的不想重復(fù)寫,那么這個(gè)時(shí)候就需要借助「default」關(guān)鍵字來幫忙我們解決這種問題。

  1. @Target(ElementType.METHOD) 
  2. @Retention(RetentionPolicy.RUNTIME) 
  3. public @interface MultiValueAnnotation { 
  4.      
  5.     String name(); 
  6.      
  7.     int num() default 0; 

我們在屬性上使用了 default 關(guān)鍵字來聲明 num 屬性的默認(rèn)值為 0 ,這樣子我們在使用上述那個(gè)注解的時(shí)候就可以不用手動(dòng)給num賦值了。

次關(guān)之造器解注

注解具有讓編譯器進(jìn)行編譯檢查的作用,但是如果沒有用來讀取注解的工具,那注解也不會(huì)比注釋更有用,起碼注釋可以讓開發(fā)人員更直觀的看到此段代碼的用處。

重回反射想要?jiǎng)?chuàng)建與使用 「注解處理器」,我們還需要借助反射機(jī)制來構(gòu)造這類工具。以下是簡單的例子:

  1. public class AnnotationHandle { 
  2.  
  3.     public static void track(Class<?> c) { 
  4.         for (Method m : c.getDeclaredMethods()) { 
  5.             MultiValueAnnotation annotation = m.getAnnotation(MultiValueAnnotation.class); 
  6.             if (annotation != null) { 
  7.                 System.out.println("name:" + annotation.name() + 
  8.                         "\n num:" + annotation.num()); 
  9.             } 
  10.         } 
  11.     } 
  12.  
  13.     public static void main(String[] args) { 
  14.         track(TestService.class); 
  15.     } 
  16.  
  17. /*  OUTPUT
  18.   name:test 
  19.    num:0 
  20. */ 

在上述例子中我們用到了兩個(gè)反射的方法:getDeclaredMethods()和getAnnotation()。

其中g(shù)etDeclaredMethods() 用來返回該類的所有方法,getAnnotation()用來獲取指定類型的注解對象。如果方法上沒有該注解則會(huì)返回 「null」 值。

注解元素可用類型

上述@MultiValueAnnotation注解中我們定義了 String類型的 「name」 和 int類型的 「num」,除此之外我們還可以使用其他類型如下:

  • 「基本類型」(「int、float、boolean等」)
  • 「String」
  • 「Class」
  • 「enum」
  • 「Annotation」
  • 「以上類型的數(shù)組」

如果使用了上面以外的其他類型,那么編譯器就會(huì)報(bào)錯(cuò)。而且要注意的是,「也不能使用基本類型的包裝類型」

默認(rèn)值的限制

上述例子中我們也看到了,我們可以在使用注解的時(shí)候給注解屬性賦值,也可以在定義注解的時(shí)候給注解一個(gè)默認(rèn)值,但是這兩者都說明了一件事:「那就是,注解元素不能有不確定的值,要么具有默認(rèn)值,要么在使用注解時(shí)提供元素的值」

基本元素不存在null值,因此對于非基本類型的元素,無論是在使用中聲明,還是在定義時(shí)聲明, 「都不能將 null 值作為其值」。因此在實(shí)際開發(fā)中,我們往往會(huì)定義一些特殊值作為不存在的標(biāo)識(shí),例如 「負(fù)數(shù)」 或 「空字符串」

三關(guān)之運(yùn)注帷幄

在前面兩關(guān)中,我們學(xué)會(huì)了定義注解和創(chuàng)建注解處理器。接下來我們就要來更加深入掌握注解!

注解也能嵌套

在修飾注解元素的時(shí)候我們看到可以使用Annotation來修飾,估計(jì)看到那的時(shí)候會(huì)覺得有點(diǎn)奇怪。在這里就來為你來揭秘。

先來看一組注解:

@Constraints

  1. @Target(ElementType.FIELD) 
  2. @Retention(RetentionPolicy.RUNTIME) 
  3. public @interface Constraints { 
  4.     boolean primaryKey() default false
  5.     boolean unique() default false

@SQLString

  1. @Target(ElementType.FIELD) 
  2. @Retention(RetentionPolicy.RUNTIME) 
  3. public @interface SQLString { 
  4.     String name() default ""
  5.     Constraints constraints() default @Constraints; 

我們在@SQLString注解中使用Constraints注解元素,并將默認(rèn)值設(shè)為@Constraints。這個(gè)時(shí)候Constraints中的值都是@Constraints注解中定義的默認(rèn)值,如果我們要使用自定義的話,做法如下:

  1. @Target(ElementType.FIELD) 
  2. @Retention(RetentionPolicy.RUNTIME) 
  3. public @interface SQLString { 
  4.     String name() default ""
  5.     Constraints constraints() default @Constraints(primaryKey = true); 

這樣子我們就可以使用自己定義的「value」

注解不支持繼承

我們不能使用extends來繼承某個(gè)@interface,但是可以通過嵌套的方式來解決這一煩惱。

AOP與注解的搭配

「AOP」 在當(dāng)今開發(fā)中我們并不陌生,那么 「AOP」 和 「注解」 能產(chǎn)生什么化學(xué)反應(yīng)呢,請看以下代碼:

@ApiLog:

  1. @Retention(RetentionPolicy.RUNTIME) 
  2. @Target({ElementType.METHOD, ElementType.TYPE}) 
  3. public @interface ApiLog { 
  4.     /** 
  5.      * 接口名稱 
  6.      */ 
  7.     String name(); 

使用:

  1. @GetMapping(value = "/getConfig"
  2. @ApiLog(name = "獲取系統(tǒng)相關(guān)配置"
  3. public Result getConfig() throws Exception { 
  4.     return sendOK(SystemService.getConfig(type)); 

Aop使用:

  1. @Aspect 
  2. @Component 
  3. public class SysLogAspect { 
  4.     @Autowired 
  5.     private LogService logService; 
  6.      
  7.     @Pointcut("@annotation(cbuc.life.annotation.ApiLog)"
  8.     public void logPointCut() {         
  9.     } 
  10.  
  11.     @Around("logPointCut()"
  12.     public Object around(ProceedingJoinPoint point) throws Throwable { 
  13.         long beginTime = System.currentTimeMillis(); 
  14.         //執(zhí)行方法 
  15.         Object result = point.proceed(); 
  16.         //執(zhí)行時(shí)長(毫秒) 
  17.         long time = System.currentTimeMillis() - beginTime; 
  18.         //保存日志 
  19.         saveSysLog(point, time); 
  20.         return result; 
  21.     } 
  22.  
  23.     private void saveSysLog(ProceedingJoinPoint joinPoint, long time) { 
  24.         MethodSignature signature = (MethodSignature) joinPoint.getSignature(); 
  25.         Method method = signature.getMethod(); 
  26.  
  27.         LogEntity log = new LogEntity(); 
  28.         ApiLog apiLog = method.getAnnotation(ApiLog.class); 
  29.         if(apiLog != null){ 
  30.             //注解上的描述 
  31.             log.setMethodDescribe(syslog.value()); 
  32.         } 
  33.  
  34.         //請求的方法名 
  35.         String className = joinPoint.getTarget().getClass().getName(); 
  36.         String methodName = signature.getName(); 
  37.         log.setMethod(className + "." + methodName + "()"); 
  38.  
  39.         //請求的參數(shù) 
  40.         Object[] args = joinPoint.getArgs(); 
  41.         String params = JSON.toJSONString(args[0]); 
  42.         log.setParams(params); 
  43.  
  44.         //獲取request 
  45.         HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest(); 
  46.  
  47.         //設(shè)置IP地址 
  48.         log.setIp(ServletUtil.getIpAddress(request)); 
  49.  
  50.         //用戶名 
  51.         String username = LoginInfo.getUsername(); 
  52.         log.setUsername(username); 
  53.  
  54.         //保存系統(tǒng)日志 
  55.         logService.save(log); 
  56.     } 

 

通過以上代碼,我們可以在期望記錄日志的方法上增加@ApiLog注解,該方法的動(dòng)作就會(huì)被記錄進(jìn)日志表,不管方法叫什么名字,類在什么位置,都可以輕松的解決,而且沒有代碼入侵!

 

責(zé)任編輯:武曉燕 來源: 小菜良記
相關(guān)推薦

2020-08-26 14:40:38

explainMySQL數(shù)據(jù)庫

2021-03-16 15:12:57

CompletableFuture機(jī)制java

2015-03-16 11:33:16

程序員代碼bug

2024-08-12 12:25:25

SpringMVC開發(fā)

2021-01-28 09:40:33

運(yùn)維監(jiān)控工具軟件

2020-09-01 14:17:03

WindowsDefender微軟

2020-05-14 08:59:28

API網(wǎng)關(guān)性能

2022-02-22 08:25:51

typeScript泛型概念泛型使用

2024-09-09 08:36:36

Java操作遠(yuǎn)程服務(wù)器

2022-03-27 22:07:35

元宇宙虛擬人IBM

2019-12-26 09:56:34

Java多線程內(nèi)部鎖

2020-03-06 10:25:10

注解Java代碼

2019-09-03 09:30:46

ss 命令SocketLinux

2023-05-16 07:15:11

架構(gòu)模型對象

2021-07-07 06:54:37

網(wǎng)頁Selenium瀏覽器

2017-02-08 19:49:03

內(nèi)存SSDDRAM

2020-12-18 09:45:33

DockerLinux命令

2020-08-03 07:54:33

神經(jīng)網(wǎng)絡(luò)通用近似定理機(jī)器學(xué)習(xí)

2020-10-30 07:09:52

Lombok架構(gòu)師

2022-12-26 18:53:00

MQ宕機(jī)倉儲(chǔ)服務(wù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號