SpringMVC实现账号只能在一处登陆

发布时间 - 2026-01-10 23:28:48    点击率:

一、问题引导

  在Web开发中,实现一个账号只能在一处登陆有两种形式:1.当某个账号在某处登陆后,如果再在其他处登陆,将前一个账号挤掉;2.当某个账号登陆后,此账号在其他设备登陆提示已经登陆,无法登陆。 正常的应用逻辑第一种应用较为广泛,因此此篇文章讨论一下第一种逻辑在spring mvc开发中一种较为简单的实现方式。

  然而在没有长连接如WebSocket或者异步请求轮询的情况下,我们之前登陆的账号只能在下一次请求(同步或异步)才能获取被挤掉的状态(如页面跳转)。 

二、实现步骤

  1.建立一个静态Map,用来存放账号和sessionID的对应关系

  2.在登陆时,校验Map中是否已存在此账号,如果不存在说明是第一次登陆,将账号和sessionID的对应关系存放到静态Map中;如果Map中存在此账号,并且sessionID和本次请求的sessionID不一致,将Map中的sessionID替换掉,因此之前登陆的账户在发送下一次非登录和校验的请求会被拦截。

  3.创建拦截器,拦截除登陆和校验url以外的所有请求。判断请求的sessionID和静态Map中此账户对应的sessionID是否一致。如果不一致,跳转到登陆页面。

三、实现代码

1.创建一个内存数据类,用于存放静态的数据,并初始化:

  public class MemoryData {
    private static Map<String, String> sessionIDMap = new HashMap<String,String>();
    public static Map<String, String> getSessionIDMap() {
      return sessionIDMap;
    }

    public static void setSessionIDMap(Map<String, String> sessionIDMap) {
      MemoryData.sessionIDMap = sessionIDMap;
    }

  }

2.创建Controller,实现校验登陆用户

  @Controller
  public class AdminController extends BaseController{

    @Autowired
    public AdminService adminService;

    /**
    * 校验登陆管理员
    * @param request
    * @param response
    * @throws IOException
    */
    @RequestMapping(value="/checkadmin")
    public void checkUserInfo(HttpServletRequest request,HttpServletResponse response) throws IOException{

      //1在数据库查找用户
      AdminBean admin = adminService.queryUserInfo(usernameS);

      //2将admin存放到Session中
      request.getSession().setAttribute("admin", admin);

      //3在sessionIDMap中存放此用户sessionID
      String sessionID = request.getRequestedSessionId();
      String user = admin.getUsername();
      if (!MemoryData.getSessionIDMap().containsKey(user)) { //不存在,首次登陆,放入Map
        MemoryData.getSessionIDMap().put(user, sessionID);
      }else if(MemoryData.getSessionIDMap().containsKey(user)&&!StringUtils.equals(sessionID, MemoryData.getSessionIDMap().get(user))){
        MemoryData.getSessionIDMap().remove(user);
        MemoryData.getSessionIDMap().put(user, sessionID);
      }

    }

  }

3.创建拦截器

  public class SingleUserInterceptor implements HandlerInterceptor {

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object arg2, Exception arg3)

        throws Exception {
      // TODO Auto-generated method stub

    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object arg2, ModelAndView arg3)
        throws Exception {
      // TODO Auto-generated method stub

    }

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object arg2) throws Exception {
      String url = request.getRequestURI();

      //如果拦截到的是登录的页面的话放行 
      if(url.indexOf("login.do")>=0||url.indexOf("checkadmin.do")>=0){
        return true;
      }
      //如果是其他请求地址,进行拦截 
      AdminBean admin = (AdminBean) request.getSession().getAttribute("admin");
      if(admin!=null){
        String sessionid = MemoryData.getSessionIDMap().get(admin.getUsername());

        //如果用户名存在放心(即登录放行) 

        if(sessionid.equals(request.getSession().getId())){
          return true;
        }else{ //如果请求的sessionID和此账号Map中存放的sessionID不一致,跳转到登陆页

          //判断如果是异步请求,设置响应头 sessionstatus为timeout,自动跳转,否则重定向
          if(request.getHeader("x-requested-with")!=null 
              && request.getHeader("x-requested-with").equalsIgnoreCase("XMLHttpRequest")){ 
            response.setHeader("sessionstatus","timeout");
            return false;
          }else{
            String indexurl=request.getContextPath()+"/login.do";
            response.sendRedirect(indexurl);
            return false;
          }
        }
      }

      //如果session中没有admin,跳转到登陆页
      request.getRequestDispatcher(request.getContextPath()+"/index.do").forward(request, response);
      return false;
    }
  }

 4.在springmvc.xml配置文件中添加拦截器

  <!--配置拦截器, 多个拦截器,顺序执行 -->
  <mvc:interceptors> 
    <mvc:interceptor>
      <mvc:mapping path="/**"/>
      <bean class="com.jiefupay.newplat.controller.SingleUserInterceptor"/>
     </mvc:interceptor> 
  </mvc:interceptors>

 四、后续

  此种方式实现一个账号只能在一处登陆是一种较简单的方法,当然也可以通过移除session的方式实现。本文皆由本人亲测实现,如有错误,欢迎指正。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# SpringMVC  # 登陆  # springmvc拦截器登录验证示例  # SpringMVC拦截器——实现登录验证拦截器的示例代码  # 详解springmvc拦截器拦截静态资源  # SpringMVC 拦截器不拦截静态资源的三种处理方式方法  # 详解利用SpringMVC拦截器控制Controller返回值  # 浅谈springMVC拦截器和过滤器总结  # 详解SpringMVC中使用Interceptor拦截器  # 详解SpringMVC拦截器(资源和权限管理)  # SpringMVC拦截器实现登录认证  # Spring MVC--拦截器实现和用户登陆例子  # 拦截器  # 跳转到  # 在此  # 能在  # 不存在  # 一处  # 跳转  # 第一种  # 的是  # 是一种  # 首次  # 多个  # 如有  # 可以通过  # 有两种  # 建立一个  # 此种  # 创建一个  # 配置文件  # 大家多多 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel怎么在Controller之外的地方验证数据  bootstrap日历插件datetimepicker使用方法  Laravel如何将应用部署到生产服务器_Laravel生产环境部署流程  谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程  如何制作公司的网站链接,公司想做一个网站,一般需要花多少钱?  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  教你用AI将一段旋律扩展成一首完整的曲子  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  如何使用 jQuery 正确渲染 Instagram 风格的标签列表  Laravel请求验证怎么写_Laravel Validator自定义表单验证规则教程  微信小程序制作网站有哪些,微信小程序需要做网站吗?  Python数据仓库与ETL构建实战_Airflow调度流程详解  Laravel如何实现数据导出到CSV文件_Laravel原生流式输出大数据量CSV【方案】  Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复  电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?  如何快速查询域名建站关键信息?  Laravel怎么判断请求类型_Laravel Request isMethod用法  昵图网官网入口 昵图网素材平台官方入口  ,交易猫的商品怎么发布到网站上去?  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  Chrome浏览器标签页分组怎么用_谷歌浏览器整理标签页技巧【效率】  如何挑选优质建站一级代理提升网站排名?  Laravel如何实现事件和监听器?(Event & Listener实战)  js实现获取鼠标当前的位置  Laravel如何使用Scope本地作用域_Laravel模型常用查询逻辑封装技巧【手册】  手机软键盘弹出时影响布局的解决方法  JS中页面与页面之间超链接跳转中文乱码问题的解决办法  php8.4header发送头信息失败怎么办_php8.4header函数问题解决【解答】  悟空识字怎么关闭自动续费_悟空识字取消会员自动扣费步骤  东莞专业网站制作公司有哪些,东莞招聘网站哪个好?  laravel怎么实现图片的压缩和裁剪_laravel图片压缩与裁剪方法  canvas 画布在主流浏览器中的尺寸限制详细介绍  制作旅游网站html,怎样注册旅游网站?  如何用JavaScript实现文本编辑器_光标和选区怎么处理  5种Android数据存储方式汇总  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  Laravel如何使用查询构建器?(Query Builder高级用法)  如何获取免费开源的自助建站系统源码?  打造顶配客厅影院,这份100寸电视推荐名单请查收  Laravel集合Collection怎么用_Laravel集合常用函数详解  Laravel中DTO是什么概念_在Laravel项目中使用数据传输对象(DTO)  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  Android自定义控件实现温度旋转按钮效果  nginx修改上传文件大小限制的方法  Laravel如何监控和管理失败的队列任务_Laravel失败任务处理与监控  打开php文件提示内存不足_怎么调整php内存限制【解决方案】  怎么用AI帮你设计一套个性化的手机App图标?  如何用5美元大硬盘VPS安全高效搭建个人网站?  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑