關於註解的定義,使用等就不說了,在這裏直接上乾貨,自定義註解相關的東西。java
元註解的做用就是註解其餘註解,通常咱們使用自定義註解時,就須要用元註解來標註咱們本身的註解,一共有四個元註解redis
java.lang.annotation提供了四種元註解,專門註解其餘的註解(在自定義註解的時候,須要使用到元註解):
@Documented –註解是否將包含在JavaDoc中
@Retention –何時使用該註解
@Target –註解用於什麼地方
@Inherited – 是否容許子類繼承該註解數組
1.)@Retention– 定義該註解的生命週期
● RetentionPolicy.SOURCE : 在編譯階段丟棄。這些註解在編譯結束以後就再也不有任何意義,因此它們不會寫入字節碼。@Override, @SuppressWarnings都屬於這類註解。
● RetentionPolicy.CLASS : 在類加載的時候丟棄。在字節碼文件的處理中有用。註解默認使用這種方式
● RetentionPolicy.RUNTIME : 始終不會丟棄,運行期也保留該註解,所以可使用反射機制讀取該註解的信息。咱們自定義的註解一般使用這種方式。ide
2.)Target – 表示該註解用於什麼地方。默認值爲任何元素,表示該註解用於什麼地方。可用的ElementType參數包括
● ElementType.CONSTRUCTOR:用於描述構造器
● ElementType.FIELD:成員變量、對象、屬性(包括enum實例)
● ElementType.LOCAL_VARIABLE:用於描述局部變量
● ElementType.METHOD:用於描述方法
● ElementType.PACKAGE:用於描述包
● ElementType.PARAMETER:用於描述參數
● ElementType.TYPE:用於描述類、接口(包括註解類型) 或enum聲明this
3.)@Documented–一個簡單的Annotations標記註解,表示是否將註解信息添加在java文檔中。spa
4.)@Inherited – 定義該註釋和子類的關係
@Inherited 元註解是一個標記註解,@Inherited闡述了某個被標註的類型是被繼承的。若是一個使用了@Inherited修飾的annotation類型被用於一個class,則這個annotation將被用於該class的子類。code
自定義註解格式:對象
public @interface 註解名 {定義體}blog
1. Annotation型定義爲@interface, 全部的Annotation會自動繼承java.lang.Annotation這一接口,而且不能再去繼承別的類或是接口.
2. 參數成員只能用public或默認(default)這兩個訪問權修飾
3. 參數成員只能用基本類型byte,short,char,int,long,float,double,boolean八種基本數據類型和String、Enum、Class、annotations等數據類型,以及這一些類型的數組.
4. 要獲取類方法和字段的註解信息,必須經過Java的反射技術來獲取 Annotation對象,由於你除此以外沒有別的獲取註解對象的方法
5. 註解也能夠沒有定義成員,繼承
因業務須要,爲了放置表單數據重複提交,好比轉帳,誤操做點了兩下,這時候給出一個操做頻繁的提示,那麼咱們能夠寫一個註解,在須要防止重複提交的地方添加上註解就ok了。代碼以下:
自定義一個註解
1 @Target(ElementType.METHOD) 2 @Retention(RetentionPolicy.RUNTIME) 3 @Documented 4 public @interface AvoidRepeatableCommit { 5 6 long timeout() default 5; 7 }
切面:
1 @Aspect 2 @Component 3 public class VoidRepeatCommitAspectJ { 4 5 Logger logger = LoggerFactory.getLogger(this.getClass()); 6 @Autowired 7 @Qualifier("redisTemplate2") 8 private RedisTemplate<String, Object> template; 9 10 @Pointcut("@annotation(com.topband.beings.aop.AvoidRepeatableCommit)") 11 public void cutPoint(){} 12 @Around("cutPoint()") 13 public Object around(ProceedingJoinPoint point) throws Throwable{ 14 logger.info("around point : {}",point); 15 HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest(); 16 String ip = ServletUtils.getIpAddr(); 17 String token = request.getHeader(SystemConstants.TOKEN_HEADER); 18 logger.info("ip:{};token:{}",ip,token); 19 //獲取註解 20 MethodSignature signature = (MethodSignature) point.getSignature(); 21 Method method = signature.getMethod(); 22 //目標類、方法 23 String className = method.getDeclaringClass().getName(); 24 String methodName = method.getName(); 25 AvoidRepeatableCommit avoidRepeatableCommit = method.getAnnotation(AvoidRepeatableCommit.class); 26 long timeout = avoidRepeatableCommit.timeout(); 27 if (timeout < 0){ 28 timeout = 5; 29 } 30 String key =SystemConstants.TOKEN_CACHE_PREFIX + token + "-" + className + "-" + methodName; 31 logger.info("key:{}; template:{}",key,template); 32 Object obj = template.opsForValue().get(key); 33 logger.info("obj:{}",obj); 34 String str = null; 35 if(obj==null){ 36 ; 37 }else{ 38 str = (String) template.opsForValue().get(key); 39 logger.info("str:{}",str); 40 } 41 // String str = redisUtil.get(key); 42 //查詢redis 中是否存在此key 無則添加,有則拋出異常 43 if (StringUtil.isEmpty(str)){ 44 String value = token + "-" + className + "-" + methodName; 45 template.opsForValue().set(key, value, 5,TimeUnit.SECONDS); 46 return point.proceed(); 47 }else { 48 ResponseObj response = new ResponseObj(); 49 response.setStatus(Defined.STATUS_ERROR); 50 response.setMessage("操做過於頻繁"); 51 return response; 52 } 53 } 54 }
而後在須要防止重複提交的地方,加上註解就能夠了。