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

前言

这段时间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 = @RequestMapping(method = RequestMethod.GET)
  • 说明:可以作用在类或者方法上,请求地址为类上的地址+方法上的地址
  • 参数绑定说明:
    如:http://localhost/user/list?userId=10&userName=tom
@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中的注解)