Shiro中session超時頁面跳轉(zhuǎn)的處理方式
session超時頁面跳轉(zhuǎn)的處理
問題描述
shiro在管理session后,在session超時會進(jìn)行跳轉(zhuǎn),這里有兩種情況需要考慮,一種是ajax方式的請求超時,一種頁面跳轉(zhuǎn)請求的超時。
本文從這兩個方面分別考慮并處理。
ajax請求超時處理
思路:通過Filter后判定,當(dāng)前是否session超時,超時判定是否是ajax請求,如果是ajax請求,則在response頭部設(shè)置session-status值,返回到前端讀取到相應(yīng)值后進(jìn)行處理
后端Filter代碼
package com.cnpc.framework.filter;
import org.apache.shiro.SecurityUtils;
import org.apache.shiro.session.Session;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* * filter過濾器,獲取項目路徑,設(shè)置ajax超時標(biāo)識
* @author billJiang QQ:475572229
*/
public class SystemFilter implements Filter {
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException,
ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
HttpServletResponse response = (HttpServletResponse) servletResponse;
System.out.println(request.getRequestURL());
String basePath = request.getContextPath();
request.setAttribute("basePath", basePath);
if (!SecurityUtils.getSubject().isAuthenticated()) {
//判斷session里是否有用戶信息
if (request.getHeader("x-requested-with") != null
&& request.getHeader("x-requested-with").equalsIgnoreCase("XMLHttpRequest")) {
//如果是ajax請求響應(yīng)頭會有,x-requested-with
response.setHeader("session-status", "timeout");//在響應(yīng)頭設(shè)置session狀態(tài)
return;
}
}
filterChain.doFilter(request, servletResponse);
}
@Override
public void destroy() {
// TODO Auto-generated method stub
}
@Override
public void init(FilterConfig arg0) throws ServletException {
// TODO Auto-generated method stub
}
}
前端通用ajax處理
注意session-status上下文部分
function ajaxPost(url, params, callback) {
var result = null;
var headers={};
headers['CSRFToken']=$("#csrftoken").val();
$.ajax({
type : 'post',
async : false,
url : url,
data : params,
dataType : 'json',
headers:headers,
success : function(data, status) {
result = data;
if(data&&data.code&&data.code=='101'){
modals.error("操作失敗,請刷新重試,具體錯誤:"+data.message);
return false;
}
if (callback) {
callback.call(this, data, status);
}
},
error : function(err, err1, err2) {
console.log("ajaxPost發(fā)生異常,請仔細(xì)檢查請求url是否正確,如下面錯誤信息中出現(xiàn)success,則表示csrftoken更新,請忽略");
console.log(err.responseText);
if(err && err.readyState && err.readyState == '4'){
var sessionstatus=err.getResponseHeader("session-status");
if(sessionstatus=="timeout"){
//如果超時就處理 ,指定要跳轉(zhuǎn)的頁面
window.location.href=basePath+"/" ;
}
else{//csrf異常
var responseBody = err.responseText;
if (responseBody) {
responseBody = "{'retData':" + responseBody;
var resJson = eval('(' + responseBody + ')');
$("#csrftoken").val(resJson.csrf.CSRFToken);
this.success(resJson.retData, 200);
}
return;
}
}
modals.error({
text : JSON.stringify(err) + '
err1:' + JSON.stringify(err1) + '
err2:' + JSON.stringify(err2),
large : true
});
}
});
return result;
}非ajax請求超時跳轉(zhuǎn)
在本試驗中,使用jquery.load方式進(jìn)行了頁面加載,并重載jquery.fn.load改寫了該方法,通過beforeSend去除了ajax標(biāo)識,由于超時返回的登錄頁面可能嵌入當(dāng)前頁面,所以需要判斷當(dāng)前獲得的頁面是否是登錄頁面,如果是登陸頁面,則再經(jīng)過一次跳轉(zhuǎn)到登陸頁(或者首頁)。
重載的jquery.fn.load方法如下,注意beforeSend和responseText.startWith部分內(nèi)容。
var _old_load = jQuery.fn.load;
jQuery.fn.load = function( url, params, callback ) {
//update for HANZO, 2016/12/22
if (typeof url !== "string" && _old_load) {
return _old_load.apply( this, arguments );
}
var selector, type, response,
self = this,
off = url.indexOf( " " );
if ( off > -1 ) {
selector = jQuery.trim( url.slice( off ) );
url = url.slice( 0, off );
}
if ( jQuery.isFunction( params ) ) {
callback = params;
params = undefined;
} else if ( params && typeof params === "object" ) {
type = "POST";
}
if ( self.length > 0 ) {
jQuery.ajax( {
url: url,
beforeSend: function( xhr ) {
xhr.setRequestHeader('X-Requested-With', {toString: function(){ return ''; }});
},
type: type || "GET",
dataType: "html",
data: params
} ).done( function( responseText ) {
//console.log(responseText);
response = arguments;
//頁面超時跳轉(zhuǎn)到首頁
if(responseText.startWith("<!--login_page_identity-->")){
window.location.href=basePath+"/";
}else{
self.html(selector ?
jQuery("<div>").append(jQuery.parseHTML( responseText )).find(selector) :
responseText);
}
} ).always( callback && function( jqXHR, status ) {
self.each( function() {
callback.apply( this, response || [ jqXHR.responseText, status, jqXHR ] );
} );
} );
}
return this;
};可通過設(shè)置session的timeout來測試結(jié)果。需要注意的是ajax請求要使用ajaxPost方法,該方法統(tǒng)一處理了超時跳轉(zhuǎn)。
一個判斷session是否過期的小技巧
Session一直是我們做web項目經(jīng)常使用的,以前沒太注意,這次又細(xì)致的看了下!
1.session其實就是一個Map
鍵=值對,通過session.getAttribute("name");獲得session中設(shè)置的參數(shù)
2.session的過期時間是從什么時候開始計算的?
是從一登錄就開始計算還是說從停止活動開始計算?
答:從session不活動的時候開始計算,如果session一直活動,session就總不會過期。
從該Session未被訪問,開始計時; 一旦Session被訪問,計時清0;
3.設(shè)置session的失效時間
a)web.xml中
<session-config> ? ? ? ? <session-timeout>30</session-timeout> ? ? </session-config> ??
b)在程序中手動設(shè)置
session.setMaxInactiveInterval(30 * 60);//設(shè)置單位為秒,設(shè)置為-1永不過期
c)tomcat也可以修改session過期時間,在server.xml中定義context時采用如下定義:
<Context path="/livsorder" docBase="/home/httpd/html/livsorder" ? defaultSessionTimeOut="3600" isWARExpanded="true" ? isWARValidated="false" isInvokerEnabled="true" ? isWorkDirPersistent="false"/> ?
4.如何判斷session過沒過期
request.getSeesion(boolean)方法,一下子讓我恍然大悟。這個方法里面?zhèn)髁艘粋€boolean值,這個值如果是true,那么如果當(dāng)前的request的session不可用,那么就創(chuàng)建新的會話,如果存在就返回當(dāng)前的會話。如果參數(shù)是false,那么在request的當(dāng)前會話不存在的時候就返回null。
這樣我們就可以很容易的聯(lián)想到這個所謂的request的當(dāng)前會話是否存在和session過期的聯(lián)系,所以我們就可以“近似地”認(rèn)為session不存在就是session過期了,那么我們就可以很容易地判斷session是否過期了。
方法如下:
if(request.getSession(false)==null)
? ?System.out.println("Session has been invalidated!");
else
? ?System.out.println("Session is active!");可能大家注意到我上面有一個“近似地”字眼,也就是說存在特別情況。
這個特殊情況就是第一次請求還沒有創(chuàng)建會話的時候,那么用這個方法返回的仍然是null,原因我想大家應(yīng)該是顯然的。
private boolean checkSession(?? ?HttpServletRequest request,
HttpServletResponse response) {
HttpSession session = request.getSession(false);
String requestURI = request.getRequestURI();
String contextPath = request.getContextPath();
?
requestURI = requestURI.substring(contextPath.length());
?
if(requestURI.equals("/") ||
requestURI.equals("/login.jsp") ||
??? ?requestURI.equals("/login.do")?? ?||
??? ?requestURI.equals(this.errorPage))
return true;
?
if(session != null?
&& session.getAttribute(this.objName) != null?
&& session.getAttribute("year") != null)
return true;
else?
return false;
}比較好的一個辦法
//request.getSession(false)==null可以近似的判斷是否過期:如果已經(jīng)過期,那么返回的是null,但是當(dāng)起一次請求,剛剛建立一個session的時候,上述方法也返回null?
//所以應(yīng)該這個做?
if(null==request.getSession(false)){?
? ?if(true==request.getSession(true).isNew()){?
? ? ? }?
else{?
System.out.println("session已經(jīng)過期");?
}?
}以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java并發(fā)系列之AbstractQueuedSynchronizer源碼分析(條件隊列)
這篇文章主要為大家詳細(xì)介紹了Java并發(fā)系列之AbstractQueuedSynchronizer源碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-02-02
使用HttpServletResponse對象獲取請求行信息
這篇文章主要介紹了使用HttpServletResponse對象獲取請求行信息,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
Java復(fù)制一個對象并且不想復(fù)制其中的空值屬性問題
這篇文章主要介紹了Java復(fù)制一個對象并且不想復(fù)制其中的空值屬性問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
Spring請求路徑帶參數(shù)URL使用注解的寫法說明
這篇文章主要介紹了Spring請求路徑帶參數(shù)URL使用注解的寫法說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
Java幸運28系統(tǒng)搭建數(shù)組的使用實例詳解
在本篇文章里小編給大家整理了關(guān)于Java幸運28系統(tǒng)搭建數(shù)組的使用實例內(nèi)容,有需要的朋友們可以參考學(xué)習(xí)下。2019-09-09

