当前位置: 移动技术网 > IT编程>开发语言>Java > SpringMVC实现账号只能在一处登陆

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

2019年07月22日  | 移动技术网IT编程  | 我要评论

一、问题引导

  在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的方式实现。本文皆由本人亲测实现,如有错误,欢迎指正。

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

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网