Idea工具中創(chuàng)建 SpringBoot工程及入門詳解
基于IDEA創(chuàng)建項(xiàng)目Module,模塊名為04-springboot-start,組id和包名為com.cy,如圖所示:
填寫module信息,如圖所示:
選擇項(xiàng)目module版本,暫時(shí)不需要自己手動(dòng)添加任何依賴,如圖所示:
填寫Module名稱,完成module創(chuàng)建,如圖所示
項(xiàng)目Module創(chuàng)建好以后,其代碼結(jié)構(gòu)分析,如圖所示:
SpringBoot 工程中由SpringBootApplication注解描述的類為啟動(dòng)入口類,例如:
package com.cy;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplicationpublic class Application {//Application.class public static void main(String[] args) {//Main Thread SpringApplication.run(Application.class, args); }}啟動(dòng)過(guò)程概要分析
SpringBoot工程啟動(dòng)時(shí)其簡(jiǎn)易初始化過(guò)程,如圖所示:
在啟動(dòng)過(guò)程中底層做了哪些事情,大致描述如下:
1)基于配置加載類(通過(guò)ClassLoader將指定位置的類讀到內(nèi)存->底層通過(guò)線程調(diào)用IO從磁盤讀取到內(nèi)存)。
2)對(duì)類進(jìn)行分析(創(chuàng)建字節(jié)碼對(duì)象-Class類型,通過(guò)反射獲取器配置信息)。
3)對(duì)于指定配置(例如由spring特定注解描述)的對(duì)象存儲(chǔ)其配置信息(借助BeanDefinition對(duì)象存儲(chǔ))。
4)基于BeanDefinition對(duì)象中class的配置構(gòu)建類的實(shí)例(Bean對(duì)象),并進(jìn)行bean對(duì)象的管理(可能會(huì)存儲(chǔ)到bean池)。
SpringBoot 快速入門分析 業(yè)務(wù)描述在項(xiàng)目Module中定義一個(gè)類,類名為DefaultCache,然后將此類對(duì)象交給Spring創(chuàng)建并管理。最后通過(guò)單元測(cè)試對(duì)類的實(shí)例進(jìn)行分析。
API設(shè)計(jì)分析基于業(yè)務(wù)描述,進(jìn)行API及關(guān)系設(shè)計(jì),如圖所示:
基于業(yè)務(wù)及API設(shè)計(jì),進(jìn)行代碼編寫,其過(guò)程如下:
第一步:定義DefaultCache類
package com.cy.pj.common.cache;import org.springframework.stereotype.Component;/** * @Component 注解描述的類,表示此類交給Spring框架管理。 */@Componentpublic class DefaultCache {}
第二步:定義DefaultCacheTests單元測(cè)試類
package com.cy.pj.common.cache;import org.junit.jupiter.api.Test;import org.springframework.beans.factory.annotation.Autowired;@SpringBootTestpublic class DefaultCacheTests { /** * @Autowired 注解描述的屬性由spring框架按照一定規(guī)則為其注入值(賦值) * 賦值過(guò)程是怎樣的? * 1)依賴查找?(請(qǐng)問查找規(guī)則是什么?) * 2)依賴注入?(需要借助什么技術(shù)?) */ @Autowired private DefaultCache defaultCache; @Test void testDefaultCache(){ System.out.println(defaultCache.toString()); //FAQ? defaultCache變量引用的對(duì)象是由誰(shuí)創(chuàng)建的,存儲(chǔ) 到了哪里?bean pool }}
第三步:運(yùn)行單元測(cè)試類進(jìn)行應(yīng)用分析
啟動(dòng)運(yùn)行單元測(cè)試方法,檢測(cè)其輸出結(jié)果,基于結(jié)果分析:
1)SpringBoot項(xiàng)目中Bean對(duì)象的構(gòu)建。
2)SpringBoot項(xiàng)目中Bean對(duì)象的獲取。
運(yùn)行過(guò)程中的BUG分析Bean類型找不到,如圖所示:
空指針異常(NullPointerExcetpion-NPE),如圖所示:
啟動(dòng)類找不到,如圖所示:
啟動(dòng)類有多個(gè),如圖所示:
NoSuchBeanDefinition異常,如圖所示:
單元測(cè)試類中的方法添加了參數(shù),如圖所示:
第一步:創(chuàng)建項(xiàng)目Module,例如名字為05-springboot-features,如圖所示:
第二步:添加業(yè)務(wù)類ObjectPool,代碼如下:
package com.cy.pj.common.pool;@Componentpublic class ObjectPool{//假設(shè)此對(duì)象為一個(gè)對(duì)象池 public ObjectPool(){//假設(shè)運(yùn)行項(xiàng)目啟動(dòng)類,此構(gòu)造方法執(zhí)行了,說(shuō)明此類對(duì)象構(gòu)建了。 Systemd.out.println('ObjectPool()') }}
思考:一般池對(duì)象有什么特點(diǎn)?
1)在JVM內(nèi)存會(huì)開辟一塊相對(duì)比較大的空間。
2)在這塊空間中存儲(chǔ)一些對(duì)象(思考基于什么存儲(chǔ)結(jié)構(gòu)進(jìn)行存儲(chǔ)-數(shù)組,鏈表,散列表)。
3)基于“享元模式”設(shè)計(jì)思想,實(shí)現(xiàn)內(nèi)存中對(duì)象的可重用性。
第三步:定義單元測(cè)試,代碼如下:
package com.cy.pj.pool;import com.cy.pj.common.pool.ObjectPool;import org.junit.jupiter.api.Test;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.test.context.SpringBootTest;@SpringBootTestpublic class ObjectPoolTests { @Autowired private ObjectPool objectPool01; @Autowired private ObjectPool objectPool02; @Test void testObjectPool01(){ System.out.println(objectPool01==objectPool02); }}延遲加載
現(xiàn)在思考一個(gè)問題,對(duì)于ObjectPool這個(gè)類,假如項(xiàng)目啟動(dòng)以后,暫時(shí)不會(huì)用到這個(gè)池對(duì)象,是否有必要對(duì)其進(jìn)行創(chuàng)建(默認(rèn)是會(huì)創(chuàng)建的)?我們知道沒必要,因?yàn)檎加脙?nèi)存。那如何在啟動(dòng)時(shí)不創(chuàng)建此類對(duì)象呢?借助Spring框架提供的延遲加載特性進(jìn)行實(shí)現(xiàn)。例如,我們可以在需要延遲加載的類上使用@Lazy注解進(jìn)行描述,代碼如下:
package com.cy.pj.common.pool;@Lazy@Componentpublic class ObjectPool{//假設(shè)此對(duì)象為一個(gè)對(duì)象池 public ObjectPool(){//假設(shè)運(yùn)行項(xiàng)目啟動(dòng)類,此構(gòu)造方法執(zhí)行了,說(shuō)明此類對(duì)象構(gòu)建了。 Systemd.out.println('ObjectPool()') }}
此時(shí),我們?cè)偃ミ\(yùn)行運(yùn)行啟動(dòng)類,檢測(cè)ObjectPool對(duì)象是否創(chuàng)建了,假如沒有創(chuàng)建,說(shuō)明延遲加載生效了。此時(shí),我們總結(jié)一下,什么對(duì)象適合使用延遲加載特性呢?大對(duì)象,稀少用(項(xiàng)目啟動(dòng)以后,暫時(shí)用不到)的對(duì)象。
注意:延遲加載并不是延遲對(duì)類進(jìn)行加載,而是在啟動(dòng)時(shí),暫時(shí)不創(chuàng)建類的實(shí)例。假如想看一下內(nèi)存中的類是否被加載了,可以通過(guò)JVM參數(shù)進(jìn)行檢測(cè),參數(shù)為-XX:+TraceClassLoading。
對(duì)象作用域分析在實(shí)際的項(xiàng)目中內(nèi)存中的對(duì)象有一些可能要反復(fù)應(yīng)用很多次,有一些可能用完以后再也不用了或者說(shuō)應(yīng)用次數(shù)很少了。對(duì)于經(jīng)常要重復(fù)使用的對(duì)象我可考慮存儲(chǔ)到池中(例如交給spring框架進(jìn)行管理),應(yīng)用次數(shù)很少的對(duì)象那就沒必要放到池中了,用完以后讓它自己銷毀就可以了。在Spring項(xiàng)目工程中為了對(duì)這樣的對(duì)象進(jìn)行設(shè)計(jì)和管理,提供了作用域特性的支持,具體應(yīng)用:
package com.cy.pj.common.pool;@Scope('singleton')@Lazy@Componentpublic class ObjectPool{//假設(shè)此對(duì)象為一個(gè)對(duì)象池 public ObjectPool(){//假設(shè)運(yùn)行項(xiàng)目啟動(dòng)類,此構(gòu)造方法執(zhí)行了,說(shuō)明此類對(duì)象構(gòu)建了。 Systemd.out.println('ObjectPool()') }}
其中,在上面的代碼中,我們使用了@Scope注解對(duì)類進(jìn)行描述,用于指定類的實(shí)例作用域。不寫@Scope默認(rèn)就是單例(singleton)作用域,這個(gè)作用域會(huì)配合延遲加載(@Lazy)特性使用,表示此類的實(shí)例在需要時(shí)可以創(chuàng)建一份并且將其存儲(chǔ)到spring的容器中(Bean池),需要的時(shí)候從池中取,以實(shí)現(xiàn)對(duì)象的可重用。假如一些對(duì)象應(yīng)用次數(shù)非常少,可以考慮不放入池中,進(jìn)而使用@Scope('prototype')作用域?qū)︻愡M(jìn)行描述,讓此類的對(duì)象何時(shí)需要何時(shí)創(chuàng)建,用完以后,當(dāng)此對(duì)象不可達(dá)了,則可以直接被GC系統(tǒng)銷毀。
對(duì)象生命周期方法程序中的每個(gè)對(duì)象都有生命周期,對(duì)象創(chuàng)建,初始化,應(yīng)用,銷毀的這個(gè)過(guò)程稱之為對(duì)象的生命周期。在對(duì)象創(chuàng)建以后要初始化,應(yīng)用完成以后要銷毀時(shí)執(zhí)行的一些方法,我們可以稱之為生命周期方法。但不見得每個(gè)對(duì)象都會(huì)定義生命周期方法。在實(shí)際項(xiàng)目中往往一些池對(duì)象通常會(huì)定義這樣的一些生命周期方法(例如連接池)。那這樣的方法在spring工程中如何進(jìn)行標(biāo)識(shí)呢?通常要借助@PostConstruct和@PreDestroy注解對(duì)特定方法進(jìn)行描述,例如:
package com.cy.pj.common.pool;@Scope('singleton')@Lazy@Componentpublic class ObjectPool{//假設(shè)此對(duì)象為一個(gè)對(duì)象池 public ObjectPool(){ Systemd.out.println('ObjectPool()') } @PostConstruct public void init(){ System.out.println('init()'); } @PreDestroy public void destory(){ System.out.println('destory()'); }}
其中:
1)@PostConstruct 注解描述的方法為生命周期初始化方法,在對(duì)象構(gòu)建以后執(zhí)行.
2)@PreDestroy 注解描述的方法為生命周期銷毀方法,此方法所在的對(duì)象,假如存儲(chǔ)到了spring容器,那這個(gè)對(duì)象在從spring容器移除之前會(huì)先執(zhí)行這個(gè)生命周期銷毀方法(prototype作用域?qū)ο蟛粓?zhí)行此方法).
SpringBoot 項(xiàng)目中的依賴注入過(guò)程分析在SpringBoot工程中,假如類與類之間存在著一定的依賴關(guān)系,Spring是如何進(jìn)行依賴注入的呢,現(xiàn)在我們就通過(guò)一個(gè)案例做一個(gè)分析。
準(zhǔn)備工作第一步:創(chuàng)建一個(gè)項(xiàng)目module,如圖所示:
第二步:?jiǎn)?dòng)運(yùn)行項(xiàng)目,檢測(cè)是否能成功啟動(dòng)
案例設(shè)計(jì)及分析為了更好理解spring框架的底層注入機(jī)制,現(xiàn)在進(jìn)行案例API設(shè)計(jì),如圖所示:
在這個(gè)案例中單元測(cè)試類CacheTests中定義一個(gè)Cache接口類型的屬性,然后由Spring框架完成對(duì)cache類型屬性值的注入。
代碼編寫及測(cè)試分析
第一步:定義Cache接口,代碼如下:
package com.cy.pj.common.cache;public interface Cache { }
第二步:定義Cache接口實(shí)現(xiàn)類SoftCache,代碼如下:
package com.cy.pj.common.cache; @Componentpublic class SoftCache implements Cache{}
第三步:定義Cache接口實(shí)現(xiàn)類WeakCache,代碼如下:
package com.cy.pj.common.cache; @Componentpublic class WeakCache implements Cache{}
第四步:定義CacheTests單元測(cè)試類,代碼如下:
package com.cy.pj.common.cache;import org.junit.jupiter.api.Test;@SpringBootTest public class CacheTests { @Autowired @Qualifier('softCache') private Cache cache; @Test public void testCache() { System.out.println(cache); }}
其中,@Autowired由spring框架定義,用于描述類中屬性或相關(guān)方法(例如構(gòu)造方法)。Spring框架在項(xiàng)目運(yùn)行時(shí)假如發(fā)現(xiàn)由他管理的Bean對(duì)象中有使用@Autowired注解描述的屬性或方法,可以按照指定規(guī)則為屬性賦值(DI)。其基本規(guī)則是:首先要檢測(cè)容器中是否有與屬性或方法參數(shù)類型相匹配的對(duì)象,假如有并且只有一個(gè)則直接注入。其次,假如檢測(cè)到有多個(gè),還會(huì)按照@Autowired描述的屬性或方法參數(shù)名查找是否有名字匹配的對(duì)象,有則直接注入,沒有則拋出異常。最后,假如我們有明確要求,必須要注入類型為指定類型,名字為指定名字的對(duì)象還可以使用@Qualifier注解對(duì)其屬性或參數(shù)進(jìn)行描述(此注解必須配合@Autowired注解使用)。
第五步:運(yùn)行CacheTests檢測(cè)輸出結(jié)果,基于結(jié)果理解其注入規(guī)則。
編寫及測(cè)試過(guò)程中的BUG分析依賴注入異常,如圖所示:
本小節(jié)為springboot技術(shù)入門章節(jié),主要講述了SpringBoot工程下,spring中bean對(duì)象的編寫,特性以及依賴注入的規(guī)則,希望通過(guò)這一小節(jié)的講解,同學(xué)們能夠理解我們?yōu)槭裁匆獙?duì)象交給spring管理,spring管理對(duì)象有什么優(yōu)勢(shì),我們?cè)趕pringboot工程中應(yīng)該如何配置這些對(duì)象。
到此這篇關(guān)于Idea工具中創(chuàng)建 SpringBoot工程及入門分析詳解的文章就介紹到這了,更多相關(guān)idea創(chuàng)建 SpringBoot工程內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!
相關(guān)文章:
1. python如何實(shí)現(xiàn)word批量轉(zhuǎn)HTML2. PHP如何打印跟蹤調(diào)試信息3. Java8內(nèi)存模型PermGen Metaspace實(shí)例解析4. XML入門的常見問題(二)5. HTML <!DOCTYPE> 標(biāo)簽6. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)7. 小技巧處理div內(nèi)容溢出8. 匹配模式 - XSL教程 - 49. 存儲(chǔ)于xml中需要的HTML轉(zhuǎn)義代碼10. HTML DOM setInterval和clearInterval方法案例詳解
