美文网首页Spring源码分析spring源码
Spring源码(八)-Spring-IOC中的注解

Spring源码(八)-Spring-IOC中的注解

作者: 阿亮私语 | 来源:发表于2017-08-24 19:11 被阅读136次

    前言

    这段时间bug有点多,白天的时间几乎都用在改bug了,近一个月都很少看书了,得赶紧改完bug看书。在spring源码分析结束之后,我们也已经完成了循环依赖的解决方案以及设计模式,那么接下来我们分析下Spring中常用的注解。
    这里先说明下注解类型。

    • Annotition的类型使用关键字 @interface 而不是interface。它继承了java.lang.annotition.Annotition接口,并非申明了一个interface。

    • Annotation类型、方法定义是独特的、受限制的:

    • Annotation类型的方法必须申明为无参数、无异常抛出的。这些方法定义了 Annotation的成员:方法名为成员名,而方法返回值为成员的类型。

    特别的,如果方法名为 value(), 则在注解的地方需要设置属性值时可以直接写入值,如:@Target({ElemenetType.TYPE}) 而不必写为 @Target(value={ElemenetType.TYPE})
    方法返回值必须为primitive类型、Class类型、枚举类型、Annotation类型或者由前面类型之一作为元素的一位数组。
    

    方法的后面可以使用default关键字加一个默认数值来申明成员的默认值,null不能作为成员的默认值,这与我们在非Annotation类型中定义方法有很大不同。
    Annotation类型和他的方法不能使用Annotation类型的参数,成员不能是generic。只有返回值类型是Class的方法可以在Annotation类型中使用generic,因为此方法能够用类转换将各种类型转换为Class。
    Annotation类型又与接口有着近似之处:它可以定义常量、静态成员类型(比如枚举类型定义);Annotation类型也可以如接口一般被实现或者继承。

    1、元注解

    元注解的作用就是负责注解其他注解。Java5.0定义了4个标准的meta-annotation类型,它们被用来提供对其它annotation类型作说明 。

    • 1.@Target,
    • 2.@Retention,
    • 3.@Documented,
    • 4.@Inherited
      这些可以通过Spirngboot的注解@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 {
    

    1.1、@Target

    用于设定注解使用范围
    【详细参考】@Target注解
    Target通过ElementType来指定注解可使用范围的枚举集合。

    • ElementType.TYPE: 接口、类、枚举、注解
    • ElementType.FIELD: 字段、枚举的常量
    • ElementType.METHOD: 方法
    • ElementType.PARAMETER: 方法参数
    • ElementType.CONSTRUCTOR: 构造函数
    • ElementType.LOCAL_VARIABLE: 局部变量
    • ElementType.ANNOTATION_TYPE: 注解
    • ElementType.PACKAGE: 包

    1.2、@Retention

    定义注解的保留策略。
    【详细参考】@Retention
    通过RetentionPolicy来指定注解的枚举集合。

    • RetentionPolicy.SOURCE :注解仅存在于源码中,在class字节码文件中不包含
    • RetentionPolicy.CLASS:默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得
    • RetentionPolicy.RUNTIME:注解会在class字节码文件中存在,在运行时可以通过反射获取到

    1.3、@Documented

    用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员。

    1.4、@Inherited

    说明子类可以继承父类中的该注解

    • 被@Inherited注解的注解只有在class上使用才会有“自动继承的特性”
    • “自动继承的特性”是指如果在子类上搜索注解,其父类上的被@Inherited注解过的注解会考虑在内
      【参考】@Inherited

    2、Spring-IOC中的注解

    2.1 、Autowired注解

    按照byType注入,这是最核心的注解,左右就不说明了,直接看源码

    @Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface Autowired {
        boolean required() default true;
    }
    

    这里我们看到了元注解,说明所有的注解都是基于这四种元注解。

    • @Autowired 根据bean 类型从spring 上下文中进行查找,注册类型必须唯一,否则报异常。与@Resource 的区别在于,@Resource 允许通过bean 名称或bean 类型两种方式进行查找@Autowired(required=false) 表示,如果spring 上下文中没有找到该类型的bean 时, 才会使用new()的方式;

    • @Autowired 标注作用于 Map 类型时,如果 Map 的 key 为 String 类型,则 Spring 会将容器中所有类型符合 Map 的 value 对应的类型的 Bean 增加进来,用 Bean 的 id 或 name 作为 Map 的 key。

    • @Autowired 还有一个作用就是,如果将其标注在 BeanFactory 类型、ApplicationContext 类型、ResourceLoader 类型、ApplicationEventPublisher 类型、MessageSource 类型上,那么 Spring 会自动注入这些实现类的实例,不需要额外的操作。

    2.2 @Qualifier

    通常和 @Autowired一起使用

    @Autowired
    @Qualifier("softService")
    private ISoftPMService softPMService;
    
    • 说明
      使用@Autowired 时,如果找到多个同一类型的bean,则会抛异常,此时可以使用 @Qualifier("beanName"),明确指定bean的名称进行注入,此时与 @Resource指定name属性作用相同。

    2.3 @Resource

    @Resource默认按照ByName自动注入
    @Resource有两个重要的属性:name和type,而Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型

    2.4、RequestMapping

    这是springMVC中最常用的注解,在Spring4.0以后的版本推出了@GetMapping,@PostMapping,@PutMapping,@DeleteMapping等,这些是为restful风格提供的封装注解,其实就是@RequestMapping和对应RequestMethod合并的注解。

    @GetMapping("/user/list")
    public User getUser(User user){
        
        ···········
        
    }
    

    这样userId和username会映射到user对应的属性上。

    2.5、@RequestParam,@PathVariable

    用于绑定参数,@PathVariable用于邦迪地址中rest风格的参数,@RequestParam用于绑定普通提交的参数。

    @PostMapping("/status/update/{id}")
    public RestResult updateStatus(@PathVariable Long id, @RequestParam("status") Boolean status) {
        
        ··········
        
    }
    
    

    @RequestParam(value="id",required=false),required = faluse该参数非必须,默认为true

    2.6、@Scope

    该方法指定bean的创建类型,默认为单例,另外scope还有prototype、request、session、global session作用域。scope="prototype"多例

    • 1.singleton : 全局有且仅有一个实例

    • 2.prototype:每次获取Bean的时候会有一个新的实例

    • 3、request:request表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP request内有效

    • 4、session:session作用域表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP session内有效

    • 5、global session作用域类似于标准的HTTPSession作用域,不过它仅仅在基于portlet的web应用中才有意义

    2.7、@Service,@Repository,@Component

    用于标注业务层组件,可以看到该注解默认使用了@Component
    【@Service】

    @Target({ElementType.TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    @Component
    public @interface Service {
      String value() default "";
    }
    
    

    【@Repository】

    @Target({ElementType.TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    @Component
    public @interface Repository {
        String value() default "";
    }
    
    

    【@Component】

    @Target(ElementType.TYPE)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface Component {
        String value() default "";
    }
    
    

    由此可见这三个注解没有功能没有太大的区别,主要是用来标注业务,做业务方面的区分。

    • @Service用于标注业务层组件

    • @Repository用于标注数据访问组件,即DAO组件

    • @Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

    2.8、@Controller、@RestController

    用于标注控制层组件即SpringMVC中的Controler
    在4.0之后推出了@RestController其实就是@Controller和@ResponseBody的合集

    2.9、@ResponseBody

    该注解用于将Controller的方法返回的对象,通过适当的HttpMessageConverter转换为指定格式后,写入到Response对象的body数据区。
    作用域类或者方法上
    【详细参考】(http://www.cnblogs.com/fangjian0423/p/springMVC-request-param-analysis.html)
    这块详细的到springMVC源码分析的时候再讲

    3.0、@RequestBody

    • 该注解用于读取Request请求的body部分数据,使用系统默认配置的HttpMessageConverter进行解析,然后把相应的数据绑定到要返回的对象上;

    • 再把HttpMessageConverter返回的对象数据绑定到 controller中方法的参数上。

    写在最后

    这一篇主要是写了一些Sping的注解,下一篇讲下配合SpingBoot推出的注解

    相关文章

      网友评论

        本文标题:Spring源码(八)-Spring-IOC中的注解

        本文链接:https://www.haomeiwen.com/subject/myoedxtx.html