@AliasFor的使用方法
创始人
2024-05-30 05:43:42
0

目录

一、@AliasFor注解概览

1、对于@AliasFor注释的翻译说明:

Explicit aliases within an annotation.

Explicit alias for attribute in meta-annotation.

Implicit aliases within an annotation.

二、@AliasFor源码属性说明

1、value属性

2、attribute属性

3、annotation属性

三、@AliasFor的使用场景

四、使用举例

1、指定别名,属性互换

①、代码举例

②、使用说明

③、 Spring中的@RequestMapping注解

2、在注解中隐示标明属性的别名

①、代码举例

3、在注解上声明了另一个注解,对另一个注解的属性进行别名覆盖

①、代码举例

②、使用说明

③、Spring中的@Configuration注解

4、组合多个注解,达到通过一个注解使用多个注解的效果

①、代码举例

②、Spring中的@SpringBootApplication注解

 五、使用@AliasFor的好处


一、@AliasFor注解概览

1、对于@AliasFor注释的翻译说明:

Explicit aliases within an annotation.

within a single annotation, @AliasFor can be declared on a pair of attributes to signal that they are interchangeable aliases for each other.

意思是注解中的属性可以互相为别名进行使用。

Explicit alias for attribute in meta-annotation.

if the annotation attribute of @AliasFor is set to a different annotation than the one that declares it, the attribute is interpreted as an alias for an attribute in a meta-annotation (i.e., an explicit meta-annotation attribute override). This enables fine-grained control over exactly which attributes are overridden within an annotation hierarchy. In fact, with @AliasFor it is even possible to declare an alias for the value attribute of a meta-annotation.

意思是注解中使用了元注解时,可以对元注解的值进行重写,目的是为了能达到和类继承中override相似的功能。

Implicit aliases within an annotation.

if one or more attributes within an annotation are declared as attribute overrides for the same meta-annotation attribute (either directly or transitively), those attributes will be treated as a set of implicit aliases for each other, resulting in behavior analogous to that for explicit aliases within an annotation.

意思是注解中使用了元注解时,可以对元注解的值进行重写,并且可用多个不同的别名进行重写,达到组合使用的效果。

二、@AliasFor源码属性说明

package org.springframework.core.annotation;import java.lang.annotation.Annotation;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
@Documented
public @interface AliasFor {@AliasFor("attribute")String value() default "";@AliasFor("value")String attribute() default "";Class annotation() default Annotation.class;
}

1、value属性

用于声明别名关系的目标注解属性名称。例如,如果我们在一个注解中使用@AliasFor注解,同时将value属性设置为另一个注解的属性名称,则这两个属性将被视为别名。

2、attribute属性

该属性与value属性类似,用于声明别名关系的目标注解属性名称。通常情况下,attribute属性与value属性的作用是相同的。

3、annotation属性

该属性用于用于指定别名的注解类型。如果没有指定annotation属性,则默认为当前注解的类型。

三、@AliasFor的使用场景

  • 在注解中一对属性上通过声明@AliasFor,进行属性互换。

  • 在注解中隐示标明属性的别名。

  • 在注解上声明了另一个注解,对另一个注解的属性进行别名覆盖(也可以理解为将一个注解上的属性值传递给另一个注解)。   

  • 组合多个注解,达到通过一个注解使用多个注解的效果。

在注解中一对属性上通过声明@AliasFor,进行属性互换构成别名对的每个属性都应该用@AliasFor注释,并且属性或值必须引用该对中的另一个属性。单从技术上看的话,从Spring Framework 5.2.1开始,允许只注释别名对中的一个属性,但是,建议在别名对中对这两个属性进行注释,以获得更好的文档以及与Spring Framework早期版本的兼容性。

四、使用举例

1、指定别名,属性互换

①、代码举例

这里定义了一个@MyAnnotation注解,其中value属性和name属性相互为别名,desc属性为普通属性。使用该注解时,可以使用value属性或者name属性设置注解的值。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyAnnotation {@AliasFor("name")String value() default "";@AliasFor("value")String name() default "";String desc() default "";
}
@MyAnnotation(name = "myName", desc = "myDescription")
public class MyClass {// ...
}

在这个例子中,@MyAnnotation(name = "myName", desc = "myDescription")和@MyAnnotation(value = "myName", desc = "myDescription")是等价的。

②、使用说明

  • 必须以属性别名对的形式出现,即要求有两个属性,且这两个属性的名字分别为对方别名。

  • 这两个属性的必须拥有相同的返回值类型。

  • 这两个属性必须拥有相同的默认值。

  • @AliasFor中的annotation()不应该被指定值。

③、 Spring中的@RequestMapping注解

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Mapping
public @interface RequestMapping {String name() default "";@AliasFor("path")String[] value() default {};@AliasFor("value")String[] path() default {};RequestMethod[] method() default {};String[] params() default {};String[] headers() default {};String[] consumes() default {};String[] produces() default {};
}

2、在注解中隐示标明属性的别名

①、代码举例

比如,Spring中的@Bean注解,我们在使用时这个注解时,可以直接填写内容,而不用主动指定"name"和"value"属性,从而达到了隐示表明属性别名的效果。

@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Bean {@AliasFor("name")String[] value() default {};@AliasFor("value")String[] name() default {};/** @deprecated */@DeprecatedAutowire autowire() default Autowire.NO;boolean autowireCandidate() default true;String initMethod() default "";String destroyMethod() default "(inferred)";
}
@Configuration
public class MyConfiguration {@Bean("myBean")public MyBean myBean() {return new MyBean();}
}

3、在注解上声明了另一个注解,对另一个注解的属性进行别名覆盖

①、代码举例

我们有一个注解@MyAnnotation,我们想要为它定义一个别名注解@MyAliasAnnotation,可以这样定义:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyAnnotation {@AliasFor("name")String value() default "";@AliasFor("value")String name() default "";String desc() default "";
}
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@MyAnnotation
public @interface MyAliasAnnotation {@AliasFor(annotation = MyAnnotation.class, attribute = "value")String name() default "";
}

在这个例子中,@MyAliasAnnotation被标记为别名注解,它使用了@MyAnnotation的属性value作为自己的name属性的别名。其中annotation属性指定了原注解的类型,attribute属性指定了要使用的属性名称。

②、使用说明

  • 在需要override的注解的属性上使用:@AliasFor(annotation="元注解",attribute="元注解的属性")。

  • 被标记@AliasFor的属性和atttibute所指向的元注解属性必须有相同的返回值类型。

  • @AliasFor中annotation指向的元注解必须作用于正在定义的注解上(就是要标记在新注解上)。

③、Spring中的@Configuration注解

package org.springframework.context.annotation;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.core.annotation.AliasFor;
import org.springframework.stereotype.Component;@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {@AliasFor(annotation = Component.class)String value() default "";
}

可以看出对于value这个属性来说,@Configuration注解中的值会重写@Component的value属性值,这有点像类之间的继承,子类可以重父类的方法。我们也可以将@Configuration注解看成@Component的子注解。

当然,你也可以理解为将一个注解上的属性值传递给另一个注解,我个人更习惯采用这种理解,因为后续可以通过一个注解组合多个注解进行使用,通过一个注解将值传递给多个注解,而这与java的继承的概念并不相同。

4、组合多个注解,达到通过一个注解使用多个注解的效果

①、代码举例

假设我们有三个注解:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation1 {String value();
}@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation2 {String name();
}@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation3 {String desc();
}

我们可以使用@AliasFor注解来定义一个新注解,将这三个注解组合起来,具体代码如下:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@MyAnnotation1
@MyAnnotation2
@MyAnnotation3
public @interface MyCombinedAnnotation {@AliasFor(annotation = MyAnnotation1.class, attribute = "value")String value() default "";@AliasFor(annotation = MyAnnotation2.class, attribute = "name")String name() default "";@AliasFor(annotation = MyAnnotation3.class, attribute = "desc")String desc() default "";
}

在这个新注解中,我们使用了@AliasFor注解,将MyAnnotation1中的value()方法和MyAnnotation2中的name()方法都映射到MyCombinedAnnotation中的value()方法中,同时将MyAnnotation3中的desc()方法映射到MyCombinedAnnotation中的desc()方法中。

这样,我们就可以通过一个注解来使用这三个注解的属性了。例如:

@MyCombinedAnnotation(value = "hello", name = "world", desc = "this is a combined annotation")
public class MyClass {//...
}

上面的代码等价于以下的代码:

@MyAnnotation1("hello")
@MyAnnotation2(name = "world")
@MyAnnotation3(desc = "this is a combined annotation")
public class MyClass {//...
}

注意,我们需要在MyCombinedAnnotation中加上@MyAnnotation1、@MyAnnotation2、@MyAnnotation3三个注解,表示这个新注解中包含了这三个注解的所有属性,可以通过@MyCombinedAnnotation来代替这三个注解的使用。同时,在MyCombinedAnnotation中定义的属性,需要使用@AliasFor注解来映射到MyAnnotation1、MyAnnotation2、MyAnnotation3中的对应属性。

②、Spring中的@SpringBootApplication注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {@Filter(type = FilterType.CUSTOM,classes = {TypeExcludeFilter.class}
), @Filter(type = FilterType.CUSTOM,classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {@AliasFor(annotation = EnableAutoConfiguration.class)Class[] exclude() default {};@AliasFor(annotation = EnableAutoConfiguration.class)String[] excludeName() default {};@AliasFor(annotation = ComponentScan.class,attribute = "basePackages")String[] scanBasePackages() default {};@AliasFor(annotation = ComponentScan.class,attribute = "basePackageClasses")Class[] scanBasePackageClasses() default {};@AliasFor(annotation = ComponentScan.class,attribute = "nameGenerator")Class nameGenerator() default BeanNameGenerator.class;@AliasFor(annotation = Configuration.class)boolean proxyBeanMethods() default true;
}

 五、使用@AliasFor的好处

使用@AliasFor注解可以为注解元素之间建立关联,提高注解的可读性和可维护性。具体好处如下:

  1. 简化注解定义:通过@AliasFor注解,可以简化注解定义中的元素声明,减少了重复代码,提高了代码的可读性和可维护性。

  2. 避免冗余:通过@AliasFor注解,可以将重复的注解元素关联起来,避免了冗余的元素声明,使代码更加简洁。

  3. 保持一致性:通过@AliasFor注解,可以将注解元素之间的关系明确化,保持注解的一致性,提高代码的可读性和可维护性。

  4. 明确注解元素的作用:通过@AliasFor注解,可以明确注解元素之间的关系,帮助开发人员更好地理解注解的作用和用法,提高开发效率和代码质量。

相关内容

热门资讯

喜欢穿一身黑的男生性格(喜欢穿... 今天百科达人给各位分享喜欢穿一身黑的男生性格的知识,其中也会对喜欢穿一身黑衣服的男人人好相处吗进行解...
发春是什么意思(思春和发春是什... 本篇文章极速百科给大家谈谈发春是什么意思,以及思春和发春是什么意思对应的知识点,希望对各位有所帮助,...
网络用语zl是什么意思(zl是... 今天给各位分享网络用语zl是什么意思的知识,其中也会对zl是啥意思是什么网络用语进行解释,如果能碰巧...
为什么酷狗音乐自己唱的歌不能下... 本篇文章极速百科小编给大家谈谈为什么酷狗音乐自己唱的歌不能下载到本地?,以及为什么酷狗下载的歌曲不是...
华为下载未安装的文件去哪找(华... 今天百科达人给各位分享华为下载未安装的文件去哪找的知识,其中也会对华为下载未安装的文件去哪找到进行解...
怎么往应用助手里添加应用(应用... 今天百科达人给各位分享怎么往应用助手里添加应用的知识,其中也会对应用助手怎么添加微信进行解释,如果能...
家里可以做假山养金鱼吗(假山能... 今天百科达人给各位分享家里可以做假山养金鱼吗的知识,其中也会对假山能放鱼缸里吗进行解释,如果能碰巧解...
四分五裂是什么生肖什么动物(四... 本篇文章极速百科小编给大家谈谈四分五裂是什么生肖什么动物,以及四分五裂打一生肖是什么对应的知识点,希...
一帆风顺二龙腾飞三阳开泰祝福语... 本篇文章极速百科给大家谈谈一帆风顺二龙腾飞三阳开泰祝福语,以及一帆风顺二龙腾飞三阳开泰祝福语结婚对应...
美团联名卡审核成功待激活(美团... 今天百科达人给各位分享美团联名卡审核成功待激活的知识,其中也会对美团联名卡审核未通过进行解释,如果能...