詳解SpringBoot中Session超時(shí)原理說明
一:前言:
最近支付后臺(tái)登錄一段時(shí)間后如果沒有任何操作,總是需要重新登錄才可以繼續(xù)訪問頁面,出現(xiàn)這個(gè)問題的原因就是session超時(shí),debug代碼后發(fā)現(xiàn)session的超時(shí)時(shí)間是1800s。也就是說當(dāng)1800秒內(nèi)沒有任何操作,session就會(huì)出現(xiàn)超時(shí)現(xiàn)象。那這個(gè)超時(shí)時(shí)間是如何設(shè)置的呢?然后該如何重新設(shè)置此超時(shí)時(shí)間呢?系統(tǒng)又如何判斷session超時(shí)的呢?接下來就一一進(jìn)行解答。
二:系統(tǒng)session超時(shí)時(shí)間如何默認(rèn)的?
說明:獲取session超時(shí)時(shí)間的方法為”request.getSession().getMaxInactiveInterval()",但是tomcat中設(shè)置超時(shí)時(shí)間的參數(shù)為“sessionTimeout”,那么他們是怎么聯(lián)系起來的呢?
第一步:加載sessionTimeout參數(shù)。
1、項(xiàng)目運(yùn)行初始化通過“@ConfigurationProperties”注解加載“org.springframework.boot.autoconfigure.web.ServerProperties”類。
//springBoot中默認(rèn)的配置文件為"application.yml"或者"application.perties"文件,也就是說server是其中的一個(gè)配置參數(shù)。 @ConfigurationProperties(prefix = "server", ignoreUnknownFields = true) public class ServerProperties implements EmbeddedServletContainerCustomizer, EnvironmentAware, Ordered { //代碼 }
2、上面類中“ServerProperties”繼承自“EmbeddedServletContainerCustomizer”接口。重寫customize方法,之后在此方法中“向上推”,即可找到“AbstractConfigurableEmbeddedServletContainer ”類。
@Override public void customize(ConfigurableEmbeddedServletContainer container) { //多個(gè)參數(shù)判斷,如果在application中沒配置的情況下都是null if (getPort() != null) { container.setPort(getPort()); } ...//n多個(gè)參數(shù)判斷, //以下的代碼就是重點(diǎn),因?yàn)槭莟omcat容器,所以以下條件為“真”,經(jīng)過一系列的查找父類或者實(shí)現(xiàn)接口即可找到抽象類“AbstractConfigurableEmbeddedServletContainer” //public class TomcatEmbeddedServletContainerFactory extends AbstractEmbeddedServletContainerFactory implements ResourceLoaderAware //public abstract class AbstractEmbeddedServletContainerFactory extends AbstractConfigurableEmbeddedServletContainer implements EmbeddedServletContainerFactory if (container instanceof TomcatEmbeddedServletContainerFactory) { getTomcat().customizeTomcat(this, (TomcatEmbeddedServletContainerFactory) container); } //以上代碼執(zhí)行完成之后,實(shí)際上已經(jīng)有對(duì)應(yīng)的session所有的默認(rèn)參數(shù),之后通過下面方法,將所有參數(shù)放入對(duì)應(yīng)的容器中。第3、4步就是設(shè)置過程 container.addInitializers(new SessionConfiguringInitializer(this.session)); }
3、在“AbstractConfigurableEmbeddedServletContainer”類中終于可以找到“超時(shí)時(shí)間”的相關(guān)設(shè)置
//重要代碼 //45行 private static final int DEFAULT_SESSION_TIMEOUT = (int) TimeUnit.MINUTES .toSeconds(30); //66行 private int sessionTimeout = DEFAULT_SESSION_TIMEOUT; @Override public void setSessionTimeout(int sessionTimeout) { this.sessionTimeout = sessionTimeout; } //171-188行 @Override public void setSessionTimeout(int sessionTimeout, TimeUnit timeUnit) { Assert.notNull(timeUnit, "TimeUnit must not be null"); this.sessionTimeout = (int) timeUnit.toSeconds(sessionTimeout); } /** * Return the session timeout in seconds. * @return the timeout in seconds */ public int getSessionTimeout() { return this.sessionTimeout; }
4、執(zhí)行第2步的”Container.addInitializers(new SessionConfiguringInitializer(this.session))“加載所有的配置參數(shù)。
public static class Session { /** * Session timeout in seconds. */ private Integer timeout; public Integer getTimeout() { return this.timeout; } //將session超時(shí)時(shí)間設(shè)置進(jìn)來 public void setTimeout(Integer sessionTimeout) { this.timeout = sessionTimeout; }
第二步:將上面的超時(shí)時(shí)間賦值給“MaxInactiveInterval”參數(shù)。
說明:既然上面tomcat需要的參數(shù)都已經(jīng)加載完成,那么接下來就會(huì)運(yùn)行tomcat,此處不做細(xì)講,直接進(jìn)入tomcat啟動(dòng)和加載參數(shù)說明。在“TomcatEmbeddedServletContainerFactory”類中的方法調(diào)用流程如下:
getEmbeddedServletContainer--》prepareContext--》configureContext--》configureSession--》getSessionTimeoutInMinutes。
1、調(diào)用configureSession設(shè)置tomcat的Session配置參數(shù)。
//以下代碼 private void configureSession(Context context) { long sessionTimeout = getSessionTimeoutInMinutes(); context.setSessionTimeout((int) sessionTimeout); Manager manager = context.getManager(); if (manager == null) { manager = new StandardManager(); //此處即為設(shè)置相應(yīng)的參數(shù)的位置。之后會(huì)調(diào)用StandardContext類的setManger(Manager)方法,在setManger中會(huì)調(diào)用"manager.setContext(this)" context.setManager(manager); } } //計(jì)算超時(shí)時(shí)間為分鐘(注意:此處會(huì)將之前的1800秒,轉(zhuǎn)換為30分鐘)??梢钥闯鲎罱K的時(shí)間結(jié)果是個(gè)整數(shù)的分鐘類型,也就是說如果設(shè)置的超時(shí)時(shí)間(單位為秒)不是60的倍數(shù),也會(huì)最終轉(zhuǎn)換為60的倍數(shù),并且最小超時(shí)時(shí)間設(shè)置的是60秒。 private long getSessionTimeoutInMinutes() { long sessionTimeout = getSessionTimeout(); if (sessionTimeout > 0) { sessionTimeout = Math.max(TimeUnit.SECONDS.toMinutes(sessionTimeout), 1L); } return sessionTimeout; }
2、最終將SessionTimeout賦值給MaxInactiveInterval。終于完成session超時(shí)時(shí)間設(shè)置。
//以下代碼 @Override public void setContext(Context context) { //省略其余設(shè)置代碼,直接重新設(shè)置Session超時(shí)時(shí)間,此時(shí)又將上面的分鐘單位轉(zhuǎn)為秒。此時(shí)終于給Sesseion設(shè)置了默認(rèn)超時(shí)時(shí)間。 if (this.context != null) { setMaxInactiveInterval(this.context.getSessionTimeout() * 60); this.context.addPropertyChangeListener(this); } }
三:如果自定義超時(shí)時(shí)間呢?
其實(shí)從上面的流程,已經(jīng)不難看出,只需要在“org.springframework.boot.autoconfigure.web.ServerProperties”類中找到對(duì)應(yīng)的Session參數(shù),初始化讓其加載上來即可完成設(shè)置。
/** * Get the session timeout. * @return the session timeout * @deprecated since 1.3.0 in favor of {@code session.timeout}. */ @Deprecated @DeprecatedConfigurationProperty(replacement = "server.session.timeout") public Integer getSessionTimeout() { return this.session.getTimeout(); }
所以在application中配置“server.session.timeout“即可,參數(shù)類型為long類型,單位為”秒“。
四:運(yùn)行程序是如何判斷session超時(shí)的?
其實(shí)很簡(jiǎn)單:只需要在每次本次同一個(gè)sessionequest請(qǐng)求的時(shí)間,和之前的請(qǐng)求時(shí)間進(jìn)行比較,發(fā)現(xiàn)兩個(gè)值的差已經(jīng)大于MaxInactiveInterval的值即可。
//判斷是否超時(shí) @Override public boolean isValid() { //省略多個(gè)條件判斷 if (maxInactiveInterval > 0) { //判斷此session空閑時(shí)間是否比maxInactiveInterval大,如果大的情況下,session就超時(shí) int timeIdle = (int) (getIdleTimeInternal() / 1000L); if (timeIdle >= maxInactiveInterval) { expire(true); } } return this.isValid; } //將上次訪問時(shí)間和當(dāng)前時(shí)間比較,拿到空閑時(shí)間值 @Override public long getIdleTimeInternal() { long timeNow = System.currentTimeMillis(); long timeIdle; if (LAST_ACCESS_AT_START) { timeIdle = timeNow - lastAccessedTime; } else { timeIdle = timeNow - thisAccessedTime; } return timeIdle; }
說明:
所以為了保證session超時(shí)時(shí)間長(zhǎng)點(diǎn),可以在application配置文件中配置“server.session.timeout”參數(shù)即可,參數(shù)單位為“秒”,如果參數(shù)不是60的整數(shù)倍,會(huì)轉(zhuǎn)換成60的整數(shù)倍(見二:系統(tǒng)如何設(shè)置超時(shí)時(shí)間、步驟二中的“1”中算法)。如不滿一分鐘,會(huì)轉(zhuǎn)換為60秒。
擴(kuò)展:
實(shí)際上也可以直接重寫EmbeddedServletContainerCustomizer的customize方法進(jìn)行賦值。
@Bean public EmbeddedServletContainerCustomizer containerCustomizer(){ return new EmbeddedServletContainerCustomizer() { @Override public void customize(ConfigurableEmbeddedServletContainer container) { container.setSessionTimeout(600);//單位為S } }; }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java中的PrintWriter 介紹_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
PrintWriter 是字符類型的打印輸出流,它繼承于Writer。接下來通過本文給大家介紹java中的 PrintWriter 相關(guān)知識(shí),感興趣的朋友一起學(xué)習(xí)吧2017-05-05Springboot獲取文件內(nèi)容如何將MultipartFile轉(zhuǎn)File
本文給大家介紹Springboot獲取文件內(nèi)容,將MultipartFile轉(zhuǎn)File方法,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2024-01-01基于多網(wǎng)卡環(huán)境下Eureka服務(wù)注冊(cè)IP的選擇問題
這篇文章主要介紹了基于多網(wǎng)卡環(huán)境下Eureka服務(wù)注冊(cè)IP的選擇問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03總結(jié)Java中線程的狀態(tài)及多線程的實(shí)現(xiàn)方式
Java中可以通過Thread類和Runnable接口來創(chuàng)建多個(gè)線程,線程擁有五種狀態(tài),下面我們就來簡(jiǎn)單總結(jié)Java中線程的狀態(tài)及多線程的實(shí)現(xiàn)方式:2016-07-07java實(shí)現(xiàn)簡(jiǎn)單的給sql語句賦值的示例
這篇文章主要介紹了java實(shí)現(xiàn)簡(jiǎn)單的給sql語句賦值的示例,需要的朋友可以參考下2014-05-05