首頁  >  文章  >  Java  >  SpringBoot對不同Bean註解的差別和使用場景是什麼

SpringBoot對不同Bean註解的差別和使用場景是什麼

WBOY
WBOY轉載
2023-05-12 13:31:061032瀏覽

對不同Bean註解的差異和使用場景

什麼是Bean?

談Bean的潛台詞是在說Spring中的Bean,我們都知道Spring中的BeanFactory,而Bean這個概念也是由此而來。在Spring中,只要一個類別能被實例化,並被Spring容器管理,這個類別就稱為一個Bean,或者SpringBean.

除此之外,我們還聽到一些其他的詞:

JavaBean、POJO、VO、DTO

這些叫法又是什麼意思?使用的場景又是什麼?

JavaBean

一個JavaBean是遵循Sun公司JavaBean規格的類別。 JavaBean可以理解為java中可以重複使用的元件,它滿足下面條件:

  • 有一個公共的缺省建構方法

  • ##這個類別的屬性使用getter和setter來訪問,並且命名遵從標準的規範

  • 這個類別可以序列化

##POJO( Plain Ordinary Object )

POJO是歷史遺留名稱,為什麼這樣講?因為POJO是用來指明該物件不同於Entity Beans

EntityBeans是EJB中的概念,而EJB在Spring出現後,就漸漸淡出了歷史的舞台。所以,當POJO在Martin Fowler提出時,就是指那些沒有實作任何EJB介面的普通java類別。而延用至今,嚴格講,所有的java類,都是POJO,因為現在沒有人在使用e​​jb這些老古董了。但是有時我們為了區分Spring Bean,可以將沒有被Spring管理的類別稱為POJO。

VO (Value Object)

VO指一個物件例如 java.lang.Integer 它持有一些數據,或資料物件。這個概念是 Martin Fowler企業應用架構中提出的概念。

DTO(Data Transfer Object)

DTO也是EJB種提出的概念,目的就是在資料傳輸時,透過直接傳輸對象,在網路中傳輸數據。

小結:

所以對我們而言,VO和DTO沒有區別(但是Martin Fowler可能用它們表示了不同的細分概念),而大多數時候,它們遵循JavaBean規範,所以它們也都是JavaBean。當然,它們都是POJO。

可以看出,它們本質上都是在指一個java對象,為了區分場景和功能,有了不同的叫法。開發中有時也會出現,Entity, Domain等。用來表示對實體的映射,或表的映射。一般可以這樣做來規範開發:

    對於Spring管理的對象,稱為Bean
  • 對應到資料表的物件實體類,稱為entity,放在entity目錄
  • 對於介面用於封裝數據,例如接受一個json入參,為了方便,定義一個物件封裝參數,可以叫dto (或pojo)放在pojo包,以表示它不是某個表的映射類別。
  • 註解@Bean @Component …等都有什麼差別?

用SpringBoot開發應用程式時,我們會用註解將物件交給Spring容器管理。這些註解包括:

@Component ,@Service, @Bean, @Controller ,@Repository

這些註解本質上,都是Spring標識,用來進行Bean的自動偵測。標註這些註解的類別會被Spring容器管理。

那為何要有這些分類,為何不使用一個註解就來搞定所有的工作?

首先這幾個註解根據語意,用在不同的層次

  • @Componet

    一般的元件

  • @Service

    是Service層元件

  • #@Bean

    這個要和@Configuration一塊使用,後邊再說

  • #@Controller

    是用在SpringMVC控制層

  • #@Repository

    是資料存取層

  • Spring這樣設計,是因為,這些註解不光是要做自動偵測。同時有不同的功能,例如@Repository註解,Spring會增加增強處理,進行相關的異常處理。

@Controller的bean會處理網路請求相關邏輯。所以你給所有的Bean都標註同一個註解,確實都會注入Spring容器,但是功能可能就會失效。

而且隨著Spring版本升級,可能會增加更多差異化處理。所以我們應該按照規範來註解。

再說到@Bean,我們知道Spring早期,還是透過xml配置Bean例如:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd 
    http://www.springframework.org/schema/context 
    http://www.springframework.org/schema/context/spring-context.xsd">
     
    <bean id="operations"    class="com.howtodoinjava.spring.beans.Operations"></bean>
    <bean id="employee"  class="com.howtodoinjava.spring.beans.Employee"></bean>
     
</beans>

現在,你可以理解@Configuration註解的類別就是一個xml設定文件,中間@Bean註解是xml中的bean節點

@Configuration
public class BeanConfiguration {
     @Bean
     public Operations operation(){
         return new Operations();
     }
     @Bean
     public Employee employee(){
         return new Employee();
     }
    
}

這兩種方式都是將@Bean註解回傳值注入Spring容器。 SpringBoot在啟動時,會掃描@Configuration註解,進行注入。

SpringBoot注入物件衝突如何解決?

好了,現在我們終於把想要的元件交給Spring容器管理。我們該如何使用?

我們可以用Spring上下文,取得需要的物件

public static void main(String[] args) {
        ApplicationContext application = SpringApplication.run(ConsumerApplication.class, args);
        application.getBean(Employee.class);
}

一般我們用@Autowire 註解,取得容器中的bean

@Autowire
private Employee employee;

有時我們在容器中需要注入一個類別的多個實例,以滿足需求。

比如一个接口的实现类有两个,如果直接通过@Component注入容器,则会报错。

如何区分?

@Component("bean01")
public class Bean01 implement AA{
    
}

@Component("bean02")
public class Bean02 implement AA{
    
}

没错,通过在注解中标识一个名称,来区分该对象的不同实例。

获取时:最终会初始化一个Bean01

@Autowire
@Qualifier("bean01")
private AA a;

这样有个问题,就是每次使用都需要显示声明@Qualifier来指定。有的场景下,我们可能想默认使用一个,其他情况再显式指定。这就涉及到@Primary

在注解时,标注了@Primary的Bean在没有指定的情况下,会默认加载。

比如:

@Component
@Primary
public class Bean01 implement AA{
    
}

@Component("bean02")
public class Bean02 implement AA{
    
}

使用时: 默认初始化的就是Bean01

@Autowire
private AA a;

SpringBoot的各种注解

@Configuration

表示当前类可以用作配置文件使用

可以在这个类中使用@bean注解来创建对象(类似于单例模式的饿汉式)。

方法中需要有返回值+使用new这个关键字

spring会把这个返回值放入spring容器中;

在后面的方法中如果要调用这个方法(@bean中有个属性name,命名name的值,在后面的@resource中使用使用按照名称注入)没有使用name这个属性的话,默认情况下@bean方法的方法名;

  • @importResource:用来导入xml文件,xml文件里面也是声明java对象,同样也是导入到spring容器中

  • @propertySource:用来导入property文件

可以和@value一起使用,@value来用读取property文件的内容;

  • @componentScan:用来指定扫描注解的位置,扫描把扫描到的注解生成对象放入spring容器中,

属性:basePackage:指定扫描到包的位置

默认情况下是扫描当前包和子包的位置

  • @SpringBootApplication

由三个主要注解组合而成:@SpringBootConfiguration+@EnableAutoConfiguration+@ComponentScan

  • @SpringBootCOnfiguration:表示这个类可以作为配置类使用;

  • @EnableAutoConfiguration:启动自动注入,把java文件配置好,直接注入到Spring容器中;

  • @ComponentScan:表示文件下的注解,用来创建对象

  • @ConfigurationProperties:使用在java类上,表示使用K-V自动注入到对应的java属性上,

参数prefix:把properties文件中对应的前缀.后面的属性对应到properties文件的属性中(使用在类上,所以在属性上可以自动赋值)

和@value是两种用法

@controller、@service、@Repository、@component

这些注解使用在java类上,componentScan会扫描这些完成对象的创建

  • @controller使用在控制层,完成接收请求参数,调用service层完成用户的请求,返回视图层给用户;

  • @Service:业务层的逻辑,调用dao层完成用户对数据库的操作,将处理结果返回给controller;

  • @Repository:使用对数据库进行持久化操作(保证用户的数据可以写入到数据库中),将处理结果返回给service层

##在SpringBoot中使用@mapper代替这个注解。用来告诉mybatis创建这个对象的动态代理对象

##@mapperScan(basePackage:指定的mapper文件的路径),使用在主启动类上,省的一个一个dao层都要使用到@mapper

  • @component:用来创建对象,但是对象没有前面三个有特殊的功能

  • @transaction:表示开启事务(一般使用在service层)有五个参数

1、传播行为 2、隔离级别 3、超时行为 4、回滚规则 5、是否只读

@RestController   = @responseBody+@controller

使用在类上:表示这个类是控制层,而且类中的所有方法加上@responseBody这个注解

以上是SpringBoot對不同Bean註解的差別和使用場景是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除