zoukankan      html  css  js  c++  java
  • springboot项目监听器不起作用

    1.使用springboot项目,现在有个需求是在添加或者修改某个菜单后,菜单会影响角色,角色影响用户。所有受影响的用户在要退出重新登录。

    自己实现的思路是这样的:写一个监听器,在收到某个特定的请求后,监听当前所有的用户,如果是受影响的用户,就销毁session,让重新登录。

      有了思路后,刚开始上网搜的是怎么在spring boot中添加监听:网上大部分的思路都一样:使用@ServletComponentScan和一个实现了HttpSessionListener的方法就可以了。(参考:https://www.cnblogs.com/liuyong1993/p/10012808.html)。但是自己按照这个配置了后,一直不起作用。启动时候能debug到这个自定义的监听里面,但是登录后缺不能实现

    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是存在的:

    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是存在不为空的。

    然后想到我们项目里面用的是shiro,会不会是shiro重写了session机制? 想到这个疑问,又上网搜索,最后通过这个发现是可以的(参考:https://blog.csdn.net/qq_34021712/article/details/80418112

    附上自己的配置:

     自定义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自定义的监听里面了。。。。

     

  • 相关阅读:
    Asp.net Core Kestrel 免费实现https
    sqlserver空间数据 + c# 实现查询附近的设备
    abp.vNext mvc版中的js和css
    asp.net core 3.x Identity
    asp.net core 3.x 授权默认流程
    Asp.Net WebApi 上传文件方法(原生js上传和JQ ajax上传)
    一个简单的.NET轻量级的ORM——Dikeko.ORM
    Mysql常见安装问题梳理(以5.6版本为例)
    Asp.netCore RESTful WebApi 小结
    初识Asp.netCore RESTful WebApi
  • 原文地址:https://www.cnblogs.com/thinkingandworkinghard/p/12423913.html
Copyright © 2011-2022 走看看