久久午夜无码,国产中文资源,Chinese熟女熟妇2乱2,高清五码d一区

當(dāng)前位置:首頁(yè) > 軟件開(kāi)發(fā) > 正文

throwable exception?SpringBoot是如何動(dòng)起來(lái)的

throwable exception?SpringBoot是如何動(dòng)起來(lái)的

各位老鐵們好,相信很多人對(duì)throwable exception都不是特別的了解,因此呢,今天就來(lái)為大家分享下關(guān)于throwable exception以及Spring...

各位老鐵們好,相信很多人對(duì)throwable exception都不是特別的了解,因此呢,今天就來(lái)為大家分享下關(guān)于throwable exception以及SpringBoot是如何動(dòng)起來(lái)的的問(wèn)題知識(shí),還望可以幫助大家,解決大家的一些困惑,下面一起來(lái)看看吧!

java web項(xiàng)目中如何優(yōu)雅的處理異常

如果Java方法不能按照正常的流程執(zhí)行,那么可以通過(guò)另外一種途徑退出:拋出一個(gè)封裝了錯(cuò)誤信息的對(duì)象,這個(gè)就是Java的異常;當(dāng)發(fā)生異常時(shí),后面的代碼無(wú)法繼續(xù)執(zhí)行,而是由異常處理器繼續(xù)執(zhí)行。

01.異常的分類

Throwable是所有異常的超類,下一級(jí)可以分為Error和Exception:

1.Error

Error是指Java運(yùn)行時(shí)系統(tǒng)內(nèi)部的錯(cuò)誤,或者說(shuō)它代表了JVM本身的錯(cuò)誤,通常都是比較嚴(yán)重的錯(cuò)誤,比如內(nèi)存溢出,虛擬機(jī)錯(cuò)誤等等;

Error通常和硬件或JVM有關(guān),和程序本身無(wú)關(guān),所以不能被代碼捕獲和處理。

2.Exception

我們經(jīng)常說(shuō)的異常是指Exception,又可以分成運(yùn)行時(shí)異常和檢查異常。

RuntimeException:運(yùn)行時(shí)異常,這類異常在編譯期間不強(qiáng)制代碼捕捉,但是可能在在JVM運(yùn)行期間拋出異常;出現(xiàn)此類異常,通常是代碼的問(wèn)題,所以需要修改程序避免這類異常。常見(jiàn)的運(yùn)行時(shí)異常,比如:NullPointerException、ClassCastException等等。

CheckedException:檢查異常,這種異常發(fā)生在編譯階段,Java編譯器會(huì)強(qiáng)制代碼去捕獲和處理此類異常;比如:ClassNotFoundException、IllegalAccessException等等。

02.異常的處理方法

捕獲異常使用try...catch語(yǔ)句,把可能發(fā)生異常的代碼放到try{...}中,然后使用catch捕獲對(duì)應(yīng)的異常;

我們也可以在代碼塊中使用Throw向上級(jí)代碼拋出異常;

在方法中使用throws關(guān)鍵字,向上級(jí)代碼拋出異常;

03.Throw和throws的區(qū)別

Throw在方法內(nèi),后面跟著異常對(duì)象;而throws是用在方法上,后面跟異常類;

Throw會(huì)拋出具體的異常對(duì)象,當(dāng)執(zhí)行到Throw的時(shí)候,方法內(nèi)的代碼也就執(zhí)行結(jié)束了;throws用來(lái)聲明異常,提醒調(diào)用方這個(gè)方法可能會(huì)出現(xiàn)這種異常,請(qǐng)做好處理的準(zhǔn)備,但是不一定會(huì)真的出現(xiàn)異常。

04.如何優(yōu)雅地處理異常

不要試圖通過(guò)異常來(lái)控制程序流程,比如開(kāi)發(fā)一個(gè)接口,正確的做法是對(duì)入?yún)⑦M(jìn)行非空驗(yàn)證,當(dāng)參數(shù)為空的時(shí)候返回“參數(shù)不允許為空”,而不應(yīng)該捕捉到空指針的時(shí)候返回錯(cuò)誤提示。

僅捕獲有必要的代碼,盡量不要用一個(gè)try...catch包住大段甚至整個(gè)方法內(nèi)所有的代碼,因?yàn)檫@樣會(huì)影響JVM對(duì)代碼進(jìn)行優(yōu)化,從而帶來(lái)額外的性能開(kāi)銷。

很多程序員喜歡catch(Exceptione),其實(shí)應(yīng)該盡可能地精確地指出是什么異常。

不要忽略異常,捕捉到異常之后千萬(wàn)不能什么也不做,要么在catch{...}中輸出異常信息,要么通過(guò)Throw或throws拋出異常,讓上層代碼處理。

盡量不要在catch{...}中輸出異常后,又向上層代碼拋出異常,因?yàn)檫@樣會(huì)輸出多條異常信息,而且它們還是相同的,這樣可能會(huì)產(chǎn)生誤導(dǎo)。

不要在finally{...}中寫(xiě)return,因?yàn)閠ry{...}在執(zhí)行return之前執(zhí)行finally{...},如果finally{...}中有return,那么將不再執(zhí)行try{...}中的return。

我將持續(xù)分享Java開(kāi)發(fā)、架構(gòu)設(shè)計(jì)、程序員職業(yè)發(fā)展等方面的見(jiàn)解,希望能得到你的關(guān)注。

SpringBoot是如何動(dòng)起來(lái)的

程序入口

SpringApplication.run(BeautyApplication.class,args);

執(zhí)行此方法來(lái)加載整個(gè)SpringBoot的環(huán)境。

1.從哪兒開(kāi)始?

SpringApplication.java

/**

*RuntheSpringapplication,creatingandrefreshinganew

*{@linkApplicationContext}.

*@paramargstheapplicationarguments(usuallypassedfromaJavamainmethod)

*@returnarunning{@linkApplicationContext}

*/

publicConfigurableApplicationContextrun(String...args){

//...

}

調(diào)用SpringApplication.java中的run方法,目的是加載SpringApplication,同時(shí)返回ApplicationContext。

2.執(zhí)行了什么?

2.1計(jì)時(shí)

記錄整個(gè)SpringApplication的加載時(shí)間!

StopWatchstopWatch=newStopWatch();

stopWatch.start();

//...

stopWatch.stop();

if(this.logStartupInfo){

newStartupInfoLogger(this.mainApplicationClass)

.logStarted(getApplicationLog(),stopWatch);

}

2.2聲明

指定java.awt.headless,默認(rèn)是true一般是在程序開(kāi)始激活headless模式,告訴程序,現(xiàn)在你要工作在Headlessmode下,就不要指望硬件幫忙了,你得自力更生,依靠系統(tǒng)的計(jì)算能力模擬出這些特性來(lái)。

privatevoidconfigureHeadlessProperty(){

System.setProperty(SYSTEM_PROPERTY_JAVA_AWT_HEADLESS,System.getProperty(

SYSTEM_PROPERTY_JAVA_AWT_HEADLESS,Boolean.toString(this.headless)));

}

2.4配置監(jiān)聽(tīng)并發(fā)布應(yīng)用啟動(dòng)事件

SpringApplicationRunListener負(fù)責(zé)加載ApplicationListener事件。

SpringApplicationRunListenerslisteners=getRunListeners(args);

//開(kāi)始

listeners.starting();

//處理所有propertysources配置和profiles配置,準(zhǔn)備環(huán)境,分為標(biāo)準(zhǔn)Servlet環(huán)境和標(biāo)準(zhǔn)環(huán)境

ConfigurableEnvironmentenvironment=prepareEnvironment(listeners,applicationArguments);

//準(zhǔn)備應(yīng)用上下文

prepareContext(context,environment,listeners,applicationArguments,printedBanner);

//完成

listeners.started(context);

//異常

handleRunFailure(context,ex,exceptionReporters,listeners);

//執(zhí)行

listeners.running(context);

getRunListeners中根據(jù)type=SpringApplicationRunListener.class去拿到了所有的Listener并根據(jù)優(yōu)先級(jí)排序。

對(duì)應(yīng)的就是META-INF/spring.factories文件中的org.springframework.boot.SpringApplicationRunListener=org.springframework.boot.context.event.EventPublishingRunListener

private<T>Collection<T>getSpringFactoriesInstances(Class<T>type,

Class<?>[]parameterTypes,Object...args){

ClassLoaderclassLoader=Thread.currentThread().getContextClassLoader();

//Usenamesandensureuniquetoprotectagainstduplicates

Set<String>names=newLinkedHashSet<>(

SpringFactoriesLoader.loadFactoryNames(type,classLoader));

List<T>instances=createSpringFactoriesInstances(type,parameterTypes,

classLoader,args,names);

AnnotationAwareOrderComparator.sort(instances);

returninstances;

}

復(fù)制代碼

在ApplicationListener中,可以針對(duì)任何一個(gè)階段插入處理代碼。

publicinterfaceSpringApplicationRunListener{

/**

*Calledimmediatelywhentherunmethodhasfirststarted.Canbeusedforvery

*earlyinitialization.

*/

voidstarting();

/**

*Calledoncetheenvironmenthasbeenprepared,butbeforethe

*{@linkApplicationContext}hasbeencreated.

*@paramenvironmenttheenvironment

*/

voidenvironmentPrepared(ConfigurableEnvironmentenvironment);

/**

*Calledoncethe{@linkApplicationContext}hasbeencreatedandprepared,but

*beforesourceshavebeenloaded.

*@paramcontexttheapplicationcontext

*/

voidcontextPrepared(ConfigurableApplicationContextcontext);

/**

*Calledoncetheapplicationcontexthasbeenloadedbutbeforeithasbeen

*refreshed.

*@paramcontexttheapplicationcontext

*/

voidcontextLoaded(ConfigurableApplicationContextcontext);

/**

*Thecontexthasbeenrefreshedandtheapplicationhasstartedbut

*{@linkCommandLineRunnerCommandLineRunners}and{@linkApplicationRunner

*ApplicationRunners}havenotbeencalled.

*@paramcontexttheapplicationcontext.

*@since2.0.0

*/

voidstarted(ConfigurableApplicationContextcontext);

/**

*Calledimmediatelybeforetherunmethodfinishes,whentheapplicationcontexthas

*beenrefreshedandall{@linkCommandLineRunnerCommandLineRunners}and

*{@linkApplicationRunnerApplicationRunners}havebeencalled.

*@paramcontexttheapplicationcontext.

*@since2.0.0

*/

voidrunning(ConfigurableApplicationContextcontext);

/**

*Calledwhenafailureoccurswhenrunningtheapplication.

*@paramcontexttheapplicationcontextor{@codenull}ifafailureoccurredbefore

*thecontextwascreated

*@paramexceptionthefailure

*@since2.0.0

*/

voidfailed(ConfigurableApplicationContextcontext,Throwableexception);

}

3.每個(gè)階段執(zhí)行的內(nèi)容

3.1listeners.starting();

在加載SpringApplication之前執(zhí)行,所有資源和環(huán)境未被加載。

3.2prepareEnvironment(listeners,applicationArguments);

創(chuàng)建ConfigurableEnvironment;將配置的環(huán)境綁定到SpringApplication中;

privateConfigurableEnvironmentprepareEnvironment(

SpringApplicationRunListenerslisteners,

ApplicationArgumentsapplicationArguments){

//Createandconfiguretheenvironment

ConfigurableEnvironmentenvironment=getOrCreateEnvironment();

configureEnvironment(environment,applicationArguments.getSourceArgs());

listeners.environmentPrepared(environment);

bindToSpringApplication(environment);

if(this.webApplicationType==WebApplicationType.NONE){

environment=newEnvironmentConverter(getClassLoader())

.convertToStandardEnvironmentIfNecessary(environment);

}

ConfigurationPropertySources.attach(environment);

returnenvironment;

}

3.3prepareContext

配置忽略的Bean;

privatevoidconfigureIgnoreBeanInfo(ConfigurableEnvironmentenvironment){

if(System.getProperty(

CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME)==null){

Booleanignore=environment.getProperty("spring.beaninfo.ignore",

Boolean.class,Boolean.TRUE);

System.setProperty(CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME,

ignore.toString());

}

}

打印日志-加載的資源

BannerprintedBanner=printBanner(environment);

根據(jù)不同的WebApplicationType創(chuàng)建Context

context=createApplicationContext();

3.4refreshContext

支持定制刷新

/**

*RegisterashutdownhookwiththeJVMruntime,closingthiscontext

*onJVMshutdownunlessithasalreadybeenclosedatthattime.

*<p>Thismethodcanbecalledmultipletimes.Onlyoneshutdownhook

*(atmax)willberegisteredforeachcontextinstance.

*@seejava.lang.Runtime#addShutdownHook

*@see#close()

*/

voidregisterShutdownHook();

3.5afterRefresh

刷新后的實(shí)現(xiàn)方法暫未實(shí)現(xiàn)

/**

*Calledafterthecontexthasbeenrefreshed.

*@paramcontexttheapplicationcontext

*@paramargstheapplicationarguments

*/

protectedvoidafterRefresh(ConfigurableApplicationContextcontext,

ApplicationArgumentsargs){

}

3.6listeners.started(context);

到此為止,SpringApplication的環(huán)境和資源都加載完畢了;發(fā)布應(yīng)用上下文啟動(dòng)完成事件;執(zhí)行所有Runner運(yùn)行器-執(zhí)行所有ApplicationRunner和CommandLineRunner這兩種運(yùn)行器

//啟動(dòng)

callRunners(context,applicationArguments);

3.7listeners.running(context);

觸發(fā)所有SpringApplicationRunListener監(jiān)聽(tīng)器的running事件方法

希望對(duì)你有幫助

好了,關(guān)于throwable exception和SpringBoot是如何動(dòng)起來(lái)的的問(wèn)題到這里結(jié)束啦,希望可以解決您的問(wèn)題哈!

蜜臀久久99静品久久久久久| 亚洲 多毛 视频| 亚洲精品无码影在线观看| 国产精品免费无码看| 小说专区亚洲欧美| 乐都县| 在线精品国产| 亚洲精品国产美女在线观看| 精品亚洲aⅴ无码一区| 亚洲无线视频一| se超碰| 亚洲另类精品久久| 久久综合激情网| 久久久精品女人| 婷婷四房综合激情五月在线| 韩国精品久久久久久久| 亚洲国产97在线精品一区| 一本色道久久综合亚州精品蜜桃| 人妻AV天堂| 一二三区黄色电影网| 老司机AV在线播放| 在线观看AV不卡网站永久| 麻豆日本视频精品| 中国少妇内射XXXX| 国产精品自拍在| 人妻夜袭| 欧美日韩香港三级| 色婷婷国产亚洲精品色窝窝| 亚洲综合成人AV无码久久天堂| 秋霞成人| 老熟妇通奸| 亚洲国产精品成人AV| 香蕉在线一区二区三区| 亚洲图 国产| 欧美亚洲日韩男同在线| 国产剧情福利AV一区二区| 四虎影视库在线观看视频| 久久亚洲精品无码GV| 日韩AV影视网| 丰满的摩洛欧美日韩二区| 成人无码精品免费视频|