Spring中存對象和取對象的方式詳解
本文我來介紹一種更簡單的存儲對象方式和取對象方式,跟上我的節(jié)奏,我們開始吧?。?/p>
透露一下:Spring中更簡單的存對象與取對象的方式是注解。
給大家插個題外話(了解即可),注解實現(xiàn)有兩種方式:1.在編譯的時候,把注解替換成相關(guān)代碼,并添加到我們原來的代碼中。2.攔截方法,當(dāng)程序執(zhí)行到某個方法時,會執(zhí)行攔截,在執(zhí)行之前或之后或之中進行一些操作(與我們寫的注解有關(guān))。
1.存儲對象(Bean)
在上一篇我們存儲對象是在配置文件中添加<bean>標(biāo)簽,但這樣其實很麻煩,我們每次都得去那添加一個。所以為了簡化,我們可以在配置文件中配置掃描路徑,然后通過注解來存儲對象。
1.1配置掃描路徑(重要?。。。?/h3>
在使用的時候,把下面這段代碼放到你的配置文件中,把<content>標(biāo)簽中的包名替換成你項目需要掃描的包名。(這個搭配注解來實現(xiàn)存儲對象)
<?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:content="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 https://www.springframework.org/schema/context/spring-context.xsd"> <content:component-scan base-package="包名"></content:component-scan> </beans>
為什么會配置這個掃描路徑呢?舉個例子吧,假設(shè)你找個人,你是從整個中國一個一個找好找還是從一個村里找好找。所以嘛,Spring也是這樣呀,你給它配置了掃描包,Spring就會從這個包里一個一個找,看哪些添加了注解(需要Spring幫忙創(chuàng)建對象的),他就會把哪些類存儲起來。
1.如果我們配置的掃描包中存在子類,里面的注解也是會被掃描的。
2.如果我創(chuàng)建的類在掃描包外面,此時我用相關(guān)注解也沒有用的,因為根本不會掃這個類!
3.那如果我的類就是在掃描包外面,但是欸,我還是想交給Spring創(chuàng)建對象,腫么辦呢?好說嘛!用我上一篇存儲Bean的方法,掃描包可以與Bean同時生效的。
4.我們設(shè)置了掃描包,沒有添加注解,這樣也是沒辦法將對象存儲到Spring中的。(兩個要搭配使用?。。。?/p>
1.2添加注解存儲對象
將對象存儲到Spring中,有兩種可以實現(xiàn),一種是類注解,一種是方法注解。
1.2.1類注解
類注解有五種:@Controller @Service @Component @Repository @Configuration。
我先來說一下,為啥有這么多類注解。簡單來說,為了讓程序員看到類注解后更直觀的知道這個類是干什么的。JavaEE上有標(biāo)準分層(至少三層):控制層,業(yè)務(wù)層,數(shù)據(jù)層。當(dāng)然也有更多層的劃分,大家可以看阿里巴巴Java開發(fā)手冊。
控制層主要用于與用戶交互,業(yè)務(wù)層主要用于處理一些業(yè)務(wù),數(shù)據(jù)層主要與數(shù)據(jù)庫進行交互。不同層干不同事,所以也就出來了不同的注解。
1.2.1.1@Controller
這個注解主要用于控制層中。代碼如下:
@Controller//通過這個注解將對象存儲到Spring中 public class StudentController { public void study() { System.out.println("控制層aaaa"); } }
1.2.1.2@Service
這個注解一般都是用于業(yè)務(wù)層的。
@Service//通過這個注解存儲對象 public class StudentService { public void eat() { System.out.println("業(yè)務(wù)層啊啊啊"); } }
1.2.1.3@Repository
這個注解用在Dao層,也即是數(shù)據(jù)層。
@Repository public class StudentMapper { public void study() { System.out.println("我是數(shù)據(jù)層"); } }
1.2.1.4@Component
這個用于哪呢?當(dāng)我們進行一些操作不屬于三層中的任何一層的時候,我們就可以用這個注解。
@Component public class Test { public void test() { System.out.println("保密功能"); } }
1.2.1.5@Configuration
這個一般都是用于配置類上。
@Configuration public class Test1 { public void set() { System.out.println("配置類"); } }
1.2.1.6五大類注解的關(guān)系
這五大類注解其實本質(zhì)都一樣,都是為了將對象存儲到Spring中。并且@Controller / @Service / @Repository / @Configuration這幾個的源碼中都存在@Component,可以認為這四個注解均為@Component的子類,或者均為@Component的擴展。
1.2.1.7通過類注解將對象存儲到Spring中的命名規(guī)則
- 當(dāng)我們創(chuàng)建的類名為比較標(biāo)準的大駝峰形式,Spring提供的對象名為首字母小寫,其余不動的形式。
- 如果類名首字母和第二個字母都為大寫,提供的對象名為原類名。
- 除了第二條外,默認情況下都是Spring提供的對象名為首字母小寫,其余不動的形式。
1.2.2方法注解
1.2.2.1使用與注意事項
方法注解為@Bean。顧名思義,該注解用到方法上。
創(chuàng)建一個User類,并且創(chuàng)建一個Users類,我們來使用一下。
//User類 public class User { private int id; private String name; private int age; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } } //Users類 @Component public class Users { @Bean public User user1() { User user = new User(); user.setAge(18); user.setName("張三"); user.setId(1); return user; } }
注意事項:
1.@Bean注解必須搭配五大類注解使用,否則會報錯的?。榱薙pring執(zhí)行的性能)
2.通過方法注解將對象存儲到Spring中命名規(guī)則為(默認情況下)對象名等于方法名。
3.方法的返回類型必須存在,且必須為一個類。
4.我們可能不同類卻存在相同方法名的方法,此時如果我們?nèi)圆捎梅椒鳛閷ο骾d來獲取對象,此時會出現(xiàn)覆蓋。最后只有一個結(jié)果,而且還不一定是我們想要拿到的那個對象。我們有兩種方法進行解決:一種是通過注解@Order注解來給其排序,值越小的優(yōu)先級越高,越先執(zhí)行。(但是也會覆蓋)最后呈現(xiàn)的結(jié)果為值最大的。另外一種是給Bean重命名。
5.當(dāng)我們對同一類型使用多個@Bean時,通過注入的方式注入對象會報錯的,因為有多個Bean了,Spring不知道用哪個了。有兩種解決方法:(1)注入的時候,使用Resource注解,并且指定對象名,通過name屬性;(2)使用@Qualifier注解的value屬性來設(shè)置注入的名稱(必須結(jié)合Autowired使用)。
1.2.2.2Bean重命名
由于上面我們提到了當(dāng)用默認的提供名字的時候可能會出現(xiàn)誤差,所以我們可以對方法上面的@Bean進行重命名。說白了也就是對存儲到Spring種的對象指定好名字。
//可以只指定一個名字 @Bean(name = "user")//這里可以用name=...也可以用value=..... public User user1() { User user = new User(); user.setAge(18); user.setName("張三"); user.setId(1); System.out.println(1); return user; } //也可以指定多個名字(用哪個都可以) @Bean(name = {"user","use","custom"})//這里可以用name=...也可以用value=..... public User user1() { User user = new User(); user.setAge(18); user.setName("張三"); user.setId(1); System.out.println(1); return user; }
但是,當(dāng)我們重命名以后,就不可以再用方法名(Spring默認提供的名字)來取出對象了,除非你將重命名后的名字設(shè)置為了方法名。
2.取出對象
我在上一篇文章中,敘述了取出對象的方法,正常來說,我們?nèi)赃@樣去取出最終對象。但是在一些類中用到另外的類的時候,我們可以有簡單方法呀,也就是對象注入。
對象注入有三種方式:屬性注入;setters方法注入;構(gòu)造方法注入。
@Autowired是先通過類型進行查找,如果發(fā)現(xiàn)有多個,然后通過名稱(根據(jù)我們定義的屬性后面,我們定義的變量名查找)查找。(裝配順序)
@Qualifier必須和@Autowired一起使用,@Qualifier通過名字來進行注入。
@Resource先根據(jù)名稱(如果有多個相同的名稱)查再根據(jù)類型查。
@Value注入普通類型屬性。
2.1屬性注入
@Controller public class StudentController { @Autowired public StudentService studentService;//在控制類中引入了另外一個類,為了更方便使用,我們這里采用對象注入方式注入對象。 public void study() { System.out.println("控制層aaaa"); } }
屬性注入就是通過@AutoWired注解來完成的。將這個注解放在當(dāng)前類中引入的另外類最為成員變量的上面即可。
優(yōu)點:很簡單。
缺點:
1.對于有final修飾的變量無法無法進行注入。
2.兼容性差,只使用于IoC容器。
3.風(fēng)險更大,更容易違背單一設(shè)計原則。(比如在服務(wù)層,我們就想處理一套業(yè)務(wù),但通過這個我們可以進行多套業(yè)務(wù))
2.2setters方法注入
也就是通過setters方法進行注入
@Controller public class StudentController { private StudentService studentService; //添加setter方法,并添加注解(注入對象) @Autowired public void setStudentService(StudentService studentService) { this.studentService = studentService; } public void study() { System.out.println("控制層aaaa"); } }
優(yōu)點:符合單一設(shè)計規(guī)則,每個Setter方法只有一個參數(shù)。
缺點:
1.不能注入不可變對象(final修飾的)
2.使用setter注入的對象可能被修改。
2.3構(gòu)造方法注入
也就是通過構(gòu)造方法來注入對象。
@Controller public class StudentController { private StudentService studentService; //通過構(gòu)造方法來注入了 @Autowired public StudentController(StudentService studentService) { this.studentService = studentService; } public void setStudentService(StudentService studentService) { this.studentService = studentService; } public void study() { System.out.println("控制層aaaa"); } }
1.如果當(dāng)前類只有這樣的一個構(gòu)造方法,@Autowired這個注解是可以省略的。
優(yōu)點:
1.可以注入不可變對象(final修飾的)為什么呢?因為被final修飾的變量必須被復(fù)制,要不直接復(fù)制,要不就是在構(gòu)造方法中復(fù)制。
2.注入對象不會被改變的,構(gòu)造方法只會執(zhí)行一次。
3.構(gòu)造方法注入可以保證完全被初始化。
4.通用性更好。
2.4另外種注入關(guān)鍵字@Resource
和@Autowired相同之處:
- 都是進行注入。
不同之處:
- @Autowired來自Spring而@Resource來自JDK
- 與@Autowired相比,@Resource支持更多參數(shù),我們可以設(shè)置name等于什么,根據(jù)名稱注入Bean。
- @Autowired可以用于三種注入方式中,而@Resource只可以用在Setter和屬性注入中。
以上就是Spring中存對象和取對象的方式詳解的詳細內(nèi)容,更多關(guān)于Spring存對象和取對象的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
spring boot 自定義規(guī)則訪問獲取內(nèi)部或者外部靜態(tài)資源圖片的方法
這篇文章主要介紹了spring boot 自定義規(guī)則訪問獲取內(nèi)部或者外部靜態(tài)資源圖片的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-01-01使用Java編寫控制JDBC連接、執(zhí)行及關(guān)閉的工具類
這篇文章主要介紹了如何使用Java來編寫控制JDBC連接、執(zhí)行及關(guān)閉的程序,包括一個針對各種數(shù)據(jù)庫通用的釋放資源的工具類的寫法,需要的朋友可以參考下2016-03-03更簡單更高效的Mybatis?Plus最新代碼生成器AutoGenerator
這篇文章主要為大家介紹了更簡單更高效的Mybatis?Plus最新代碼生成器AutoGenerator使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02Spring boot基于JPA訪問MySQL數(shù)據(jù)庫的實現(xiàn)
本文主要介紹了Spring boot基于JPA訪問MySQL數(shù)據(jù)庫的實現(xiàn),Spring boot結(jié)合Jpa 能夠簡化創(chuàng)建 JPA 數(shù)據(jù)訪問層和跨存儲的持久層功能,用戶的持久層Dao接口只需要繼承定義好的接口,感興趣的可以了解一下2021-06-06SpringBoot+SpringSecurity實現(xiàn)認證的流程詳解
這篇文章主要介紹了SpringBoot+SpringSecurity實現(xiàn)認證的流程,文中通過代碼示例和圖文結(jié)合的方式講解的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-05-05maven關(guān)于pom文件中的relativePath標(biāo)簽使用
在Maven項目中,子工程通過<relativePath>標(biāo)簽指定父工程的pom.xml位置,以確保正確繼承父工程的配置,這個標(biāo)簽可以配置為默認值、空值或自定義值,默認情況下,Maven會向上一級目錄尋找父pom;若配置為空值2024-09-09