Java Spring

Spring 中的一些注解

1. @Component@Bean 的区别是什么?

  1. 作用对象不同:@Component 注解作用于类,而 @Bean 注解作用于方法、
  2. @Component 通常是通过路径扫描来自动侦测以及自动装配到 Spring 容器中(可以使用 @ComponentScan 注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 容器中)。@Bean 注解通常是在标有该注解的方法中定义产生这个 bean,@Bean 告诉了 Spring 这是某个类的实例,当需要用它的时候还给我。
  3. @Bean 注解比 @Component 注解的自定义性更强,而且很多地方只能通过 @Bean 注解来注册 bean。比如当引用第三方库中的类需要装配到 Spring 容器时,只能通过 @Bean 来实现。

@Bean 注解使用示例:

  1. @Configuration
  2. public class AppConfig {
  3. @Bean
  4. public TransferService transferService() {
  5. return new TransferServiceImpl();
  6. }
  7. }

@Component 注解使用示例:

  1. @Component
  2. public class ServiceImpl implements AService {
  3. ....
  4. }

下面这个例子是通过 @Component 无法实现的:

  1. @Bean
  2. public OneService getService(status) {
  3. case (status) {
  4. when 1:
  5. return new serviceImpl1();
  6. when 2:
  7. return new serviceImpl2();
  8. when 3:
  9. return new serviceImpl3();
  10. }
  11. }

2. @Autowire@Resource 的区别

  1. @Autowire@Resource都可以用来装配bean,都可以用于字段或setter方法。
  2. @Autowire 默认按类型装配,默认情况下必须要求依赖对象必须存在,如果要允许 null 值,可以设置它的 required 属性为 false。
  3. @Resource 默认按名称装配,当找不到与名称匹配的 bean 时才按照类型进行装配。名称可以通过 name 属性指定,如果没有指定 name 属性,当注解写在字段上时,默认取字段名,当注解写在 setter 方法上时,默认取属性名进行装配。 :::tips 注意:如果 name 属性一旦指定,就只会按照名称进行装配。 ::: @Autowire@Qualifier配合使用效果和@Resource一样: ```java @Autowired(required = false) @Qualifier(“example”) private Example example;

@Resource(name = “example”) private Example example;

  1. `@Resource` 装配顺序
  2. 1. 如果同时指定 name type,则从容器中查找唯一匹配的 bean 装配,找不到则抛出异常;
  3. 1. 如果指定 name 属性,则从容器中查找名称匹配的 bean 装配,找不到则抛出异常;
  4. 1. 如果指定 type 属性,则从容器中查找类型唯一匹配的 bean 装配,找不到或者找到多个抛出异常;
  5. 1. 如果不指定,则自动按照 byName 方式装配,如果没有匹配,则回退一个原始类型进行匹配,如果匹配则自动装配。
  6. <a name="FnS4d"></a>
  7. ### 3. 将一个类声明为 Spring 的 bean 的注解有哪些?
  8. - `@Component` :通用的注解,可标注任意类为 Spring 的组件。如果一个 Bean 不知道属于哪个层,可以使用 `@Component` 注解标注。
  9. - `@Repository` :对应持久层即 Dao 层,主要用于数据库相关操作。
  10. - `@Service` :对应服务层,主要设计一些复杂的逻辑,需要用到 Dao 层。
  11. - `@Controller` :对应 Spring MVC 控制层,主要用来接受用户请求并调用 Service 层返回数据给前端页面。
  12. - `@Configuration` :声明该类为一个配置类,可以在此类中声明一个或多个 `@Bean` 方法。
  13. <a name="XyEct"></a>
  14. ### 4. `@Configuration` :配置类注解
  15. `@Configuration` 表明在一个类里可以声明一个或多个 `@Bean` 方法,并且可以由 Spring 容器处理,以便在运行时为这些 bean 生成 bean 定义和服务请求,例如:
  16. ```java
  17. @Configuration
  18. public class AppConfig {
  19. @Bean
  20. public MyBean myBean() {
  21. // instantiate, configure and return bean ...
  22. }
  23. }

可以通过 AnnotationConfigApplicationContext 来注册 @Configuration 类:

  1. AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext();
  2. ctx.register(AppConfig.class);
  3. ctx.refresh();
  4. MyBean myBean = ctx.getBean(MyBean.class);
  5. // use myBean ...

另外也可以通过组件扫描(component scanning)来加载,@Configuration 使用 @Component 进行原注解,因此 @Configuration 类也可以被组件扫描到(特别是使用 XML 的 元素)。@Configuration 类不仅可以使用组件扫描进行引导,还可以使用 @ComponentScan 注解自行配置组件扫描:

  1. @Configuration
  2. @ComponentScan("com.acme.app.services")
  3. public class AppConfig {
  4. // various @Bean definitions ...
  5. }

使用 @Configuration 的约束:

  • 配置类必须以类的方式提供(比如不能是由工厂方法返回的实例)。
  • 配置类必须是非 final 的。
  • 配置类必须是非本地的(即可能不在方法中声明),native 标注的方法。
  • 任何嵌套的配置类必须声明为 static
  • @Bean 方法可能不会反过来创建更多的配置类。

除了单独使用 @Configuration 注解,还可以结合一些外部的 bean 或者注解共同使用,比如 Environment API,@PropertySource@Value@Profile 等等许多,这里就不做详细介绍了,更多的用法可以参看 Spring @Configuration 的相关文档 。

5. @ControllerAdvice :处理全局异常利器

在 Spring 3.2 中,新增了 @ControllerAdvice@RestControllerAdvice@RestController 注解,可以用于定义 @ExceptionHandler@InitBinder@ModelAttribute,并应用到所有 @RequestMapping@PostMapping@GetMapping等这些 Controller 层的注解中。
默认情况下,@ControllerAdvice 中的方法应用于全局所有的 Controller。而使用选择器 annotations()basePackageClasses()basePackages() (或其别名value())来定义更小范围的目标 Controller 子集。如果声明了多个选择器,则应用 OR 逻辑,这意味着所选的控制器应匹配至少一个选择器。请注意,选择器检查是在运行时执行的,因此添加许多选择器可能会对性能产生负面影响并增加复杂性。
@ControllerAdvice 最常使用的是结合 @ExceptionHandler 用于全局异常的处理。可以结合以下例子,可以捕获自定义的异常进行处理,并且可以自定义状态码返回:

  1. @ControllerAdvice("com.developlee.errorhandle")
  2. public class MyExceptionHandler {
  3. /**
  4. * 捕获CustomException
  5. * @param e
  6. * @return json格式类型
  7. */
  8. @ResponseBody
  9. @ExceptionHandler({CustomException.class}) //指定拦截异常的类型
  10. @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR) //自定义浏览器返回状态码
  11. public Map>String, Object< customExceptionHandler(CustomException e) {
  12. Map&lt;String, Object&gt; map = new HashMap&lt;&gt;();
  13. map.put("code", e.getCode());
  14. map.put("msg", e.getMsg());
  15. return map;
  16. }
  17. }

更多信息可以参看 Spring @ControllerAdvice 的官方文档。

6. @Component, @Repository, @Service 的区别

2021-08-21-18-39-38-040668.png
@Component是一个通用的Spring容器管理的单例bean组件。而@Repository, @Service, @Controller就是针对不同的使用场景所采取的特定功能化的注解组件。
因此,当一个类被@Component所注解,那么就意味着同样可以用@Repository, @Service, @Controller 来替代它,同时这些注解会具备有更多的功能,而且功能各异。
最后,如果不知道要在项目的业务层采用@Service还是@Component注解。那么,@Service是一个更好的选择。