1.使用springboot項目,如今有個需求是在添加或者修改某個菜單後,菜單會影響角色,角色影響用戶。全部受影響的用戶在要退出從新登陸。html
本身實現的思路是這樣的:寫一個監聽器,在收到某個特定的請求後,監聽當前全部的用戶,若是是受影響的用戶,就銷燬session,讓從新登陸。java
有了思路後,剛開始上網搜的是怎麼在spring boot中添加監聽:網上大部分的思路都同樣:使用@ServletComponentScan和一個實現了HttpSessionListener的方法就能夠了。(參考:http://www.javashuo.com/article/p-zrvbmaxl-do.html)。可是本身按照這個配置了後,一直不起做用。啓動時候能debug到這個自定義的監聽裏面,可是登陸後缺不能實現web
sessionCreated()
package com.listener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /** * session監聽器 * @author Administrator */ @WebListener public class SessionListener implements HttpSessionListener{ private int onlineCount = 0;//記錄session的數量 /** * session建立後執行 */ @Override public void sessionCreated(HttpSessionEvent se) { onlineCount++; System.out.println("【HttpSessionListener監聽器】 sessionCreated, onlineCount:" + onlineCount); se.getSession().getServletContext().setAttribute("onlineCount", onlineCount); } /** * session失效後執行 */ @Override public void sessionDestroyed(HttpSessionEvent se) { if (onlineCount > 0) { onlineCount--; } System.out.println("【HttpSessionListener監聽器】 sessionDestroyed, onlineCount:" + onlineCount); se.getSession().getServletContext().setAttribute("onlineCount", onlineCount); } } 複製代碼
還問了羣裏的大神幫忙看了下,也沒問題。剛開始懷疑是 不是登陸時候監聽的HttpSession,由於實現的是HttpSessionListener,是須要有個發起的動做的.可是本身登陸時候也有httpSession。而後在本身的測試類裏面進行測試,發現sesionId是存在的:spring
package com.sq.transportmanage.gateway.api.auth; import com.alibaba.fastjson.JSONObject; import com.sq.transportmanage.gateway.api.web.interceptor.AjaxResponse; import com.sq.transportmanage.gateway.api.web.interceptor.LoginoutListener; import com.sq.transportmanage.gateway.service.common.shiro.session.RedisSessionDAO; import com.sq.transportmanage.gateway.service.common.web.RestErrorCode; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.util.StringUtils; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import org.springframework.web.bind.annotation.RestController; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpSession; /** * @Author fanht * @Description * @Date 2020/3/5 下午6:46 * @Version 1.0 */ @RestController @RequestMapping("/loginoutController") public class LoginoutController extends RedisSessionDAO{ private Logger logger = LoggerFactory.getLogger(this.getClass()); @RequestMapping("/userLoginOut") @ResponseBody public AjaxResponse userLoginOut(String userIds, HttpSession httpSession, HttpServletRequest request){ logger.info("httpSessionId" + httpSession.getId() + ",是不是session會話:" + request.getSession(false)); HttpSession session = request.getSession(); String loginName = (String) session.getAttribute("loginName"); logger.info("loginName:" + loginName); logger.info("調用退出接口並清除shiro緩存" + userIds); logger.info("獲取監聽存取的信息" + JSONObject.toJSONString(LoginoutListener.sessionCount)); try { String userId[] = StringUtils.tokenizeToStringArray(userIds,","); for(int i = 0;i<userId.length;i++){ clearRelativeSession(null,null,Integer.valueOf(userId[i])); } return AjaxResponse.success(null); } catch (NumberFormatException e) { e.printStackTrace(); logger.error("shiro退出異常" + e); return AjaxResponse.fail(RestErrorCode.UNKNOWN_ERROR); } } @Override public void clearRelativeSession(Integer permissionId, Integer roleId, Integer userId) { super.clearRelativeSession(null, null, userId); } }
是可以打印sessionId的,也就是說session是存在不爲空的。apache
而後想到咱們項目裏面用的是shiro,會不會是shiro重寫了session機制? 想到這個疑問,又上網搜索,最後經過這個發現是能夠的(參考:https://blog.csdn.net/qq_34021712/article/details/80418112)json
附上本身的配置:api
自定義shiroSessionListener:緩存
package com.sq.transportmanage.gateway.api.web.interceptor; import com.google.common.collect.Maps; import com.sq.transportmanage.gateway.service.common.shiro.session.RedisSessionDAO; import org.apache.shiro.session.Session; import org.apache.shiro.session.SessionListener; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.Map; import java.util.concurrent.atomic.AtomicInteger; /** * @Author fanht * @Description 監聽當前有哪些用戶,當收到特定通知後通知退出登陸 * @Date 2020/3/5 下午1:48 * @Version 1.0 */ //@WebListener public class LoginoutListener extends RedisSessionDAO implements SessionListener { private Logger logger = LoggerFactory.getLogger(this.getClass()); public static final Map<Long,String> mapUser = Maps.newHashMap(); public final static AtomicInteger sessionCount = new AtomicInteger(0); @Override public void onStart(Session session) { //會話建立,在線人數加一 logger.info("======" + sessionCount); sessionCount.incrementAndGet(); } @Override public void onStop(Session session) { //會話退出,在線人數減一 sessionCount.decrementAndGet(); } @Override public void onExpiration(Session session) { //會話過時,在線人數減一 sessionCount.decrementAndGet(); } /** * 獲取在線人數使用 * @return */ public AtomicInteger getSessionCount() { return sessionCount; } /*@Override public void sessionCreated(HttpSessionEvent se) { onlineCount++; logger.info("建立start====== ===" + se.getSession().getId()); mapUser.put(se.getSession().getCreationTime(),se.getSession().getId()); } @Override public void sessionDestroyed(HttpSessionEvent se) { logger.info("銷燬session============="); }*/ }
ShiroConfiguration裏面添加配置的監聽:
@Bean("sessionManager") public DefaultWebSessionManager sessionManager(RedisSessionDAO sessionDAO, SimpleCookie sessionIdCookie) { DefaultWebSessionManager sessionManager = new DefaultWebSessionManager(); //session存活時間60分鐘 sessionManager.setGlobalSessionTimeout(3600000); sessionManager.setDeleteInvalidSessions(true); //自定義監聽 fht 不能使用@WebListern的 HttpSessionListerner 由於shiro重寫了session 2020-03-05 Collection<SessionListener> sessionListeners = new ArrayList<>(); sessionListeners.add(sessionListener()); sessionManager.setSessionListeners(sessionListeners); //sessionManager.setSessionValidationSchedulerEnabled(true); //sessionManager.setSessionValidationScheduler(sessionValidationScheduler); sessionManager.setSessionDAO(sessionDAO); sessionManager.setSessionIdCookieEnabled(true); sessionManager.setSessionIdCookie(sessionIdCookie); return sessionManager; }
/** * 自定義shiro監聽 * @return */ @Bean("sessionListener") public LoginoutListener sessionListener(){ LoginoutListener loginoutListener = new LoginoutListener(); return loginoutListener; }
而後從新啓動,測試 ,發現能夠進入到shiro自定義的監聽裏面了。。。。springboot