//登录页面 login.jsp


<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<script type="text/javascript" src="./js/jquery-3.5.1.min.js"></script>
<script type="text/javascript" src="./js/vue.js"></script>
<title>登录</title>
<%
String msg = (String)(request.getAttribute("msg")==null?"":request.getAttribute("msg"));
%>
<script type="text/javascript">
function show(){
if('<%=msg%>'!=''){
alert('<%=msg%>');
}
}
</script>
</head>
<body style="text-align: center" οnlοad="show();">
<form action="${pageContext.request.contextPath}/login.do" method="post">
<table border="1">
<tr>
<td align="center" colspan="2">
<h3>用户登录</h3>
</td>
</tr>
<tr>
<td align="right">
用户名
</td>
<td align="left">
<input type="text" id="userName" name="userName">
</td>
</tr>
<tr>
<td align="right">
密 码
</td>
<td align="left">
<input type="password" id="password" name="password">
</td>
</tr>
<tr>
<td align="center" colspan="2">
<input type="submit" value="登录"> <input type="reset" value="重置">
</td>
</tr>
</table>
</form>
</body>
</html>

  //主页面 main.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>登录成功</title>
<script type="text/javascript" src="./js/jquery-3.5.1.min.js"></script>
<script type="text/javascript" src="./js/vue.js"></script>
</head>
<%
response.setHeader("Pragma","No-cache");
response.setHeader("Cache-Control","No-cache");
response.setDateHeader("Expires", -1);
response.setHeader("Cache-Control", "No-store");
String userName = "";
userName = (String)(session.getAttribute("userName")==null?"":session.getAttribute("userName"));
%>
<script type="text/javascript">
var count;
var userMsg = '';
var i =0;
function checkUserOnline(){
$.ajax({
type:"post",
url:"${pageContext.request.contextPath}/checkUserOnline.do",
dataType : "text",
success:function(data){
userMsg = data;
//alert(JSON.stringify(userMsg))
},
error:function(){
alert("获取用户信息失败!");
clearInterval(count);
reLogin();
}
});
if(userMsg=='null'||userMsg==''|| userMsg == 'undefined'){
return;
}else{
//alert(JSON.stringify(userMsg));
clearInterval(count);
reLogin();
}
}
function reLogin(){
window.location = "login.jsp";
} function checkLogin(){
alert("检查");
count = setInterval("checkUserOnline()",5000);
}
</script>
<body onclick="checkLogin()">
<%
if(!"".equals(userName)){
out.print("登陆成功!<br/>用户名:<span id='userName'>"+userName+"</span><br/><input type='button' value='重新登陆' οnclick='reLogin();'/>");
}
%>
</body>
</html>

  //控制类 LoginController

package com.zjn.oneLogin.dengLu;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody; @Controller
public class LoginController{
/**
* 用户和Session绑定关系
*/
public static final Map<String, HttpSession> USER_SESSION = new HashMap<String, HttpSession>();
/**
* seeionId和用户的绑定关系
*/
public static final Map<String, String> SESSIONID_USER = new HashMap<String, String>();
@RequestMapping("/login")
@ResponseBody
public Map<String,Object>toLogin(HttpServletRequest request, HttpServletResponse response) throws Exception{
//获取请求命令
request.setCharacterEncoding("utf-8");
String servletPath = request.getServletPath();//获取请求路径
String uri = servletPath.substring(1, servletPath.lastIndexOf(".do")); try{
//登录
if ("login".equals(uri)){
HttpSession session = request.getSession();
String userName = request.getParameter("userName");
String password = request.getParameter("password"); if (userName != null && !"".equals(userName.trim())){
//登录成功
if (login(userName, password)){
//处理用户登录(保持同一时间同一账号只能在一处登录)
userLoginHandle(request); //添加用户与HttpSession的绑定
USER_SESSION.put(userName.trim(), session);
//添加sessionId和用户的绑定
SESSIONID_USER.put(session.getId(), userName);
System.out.println("添加sessionId和用户的绑定 ==="+session.getId());
System.out.println("用户[" + userName + "] 已上线...");
session.setAttribute("userName", userName);
session.removeAttribute("userMsg");//从session中移除用户信息
//重定向到首页
response.sendRedirect("main.jsp"); }
//登录失败
else{
System.out.println("用户[" + userName + "] 登录失败...");
request.setAttribute("msg", "登录失败,请重新登录!");
//response.sendRedirect("login.jsp");
request.getRequestDispatcher("login.jsp").forward(request, response);
}
}
else{
System.out.println("用户[" + userName + "] 登录失败..."); request.setAttribute("msg", "登录失败,请重新登录!"); //response.sendRedirect("login.jsp");
request.getRequestDispatcher("login.jsp").forward(request, response);//跳转到用户页面,返回请求响应对象
}
}else if ("reLogin".equals(uri)){
//重新登陆
HttpSession session = request.getSession();
String userName = (String)session.getAttribute("userName");
if (session != null){
//销毁相关session
//USER_SESSION.remove(SESSIONID_USER.remove(session.getId()));
session.invalidate();
}
if (userName != null && !"".equals(userName)){
System.out.println("用户[" + userName + "] 已下线...");
}
//重定向到登录页面
response.sendRedirect("login.jsp");
} }
catch (Exception e){
System.out.println(e.getClass() + e.getMessage());
PrintWriter out = response.getWriter();
out.print("服务器内部错误!");
}
return null; } /**
*
* Description:用户登录时的处理 <br>
* @param request
* @see
*/
private void userLoginHandle(HttpServletRequest request){
//当前登录的用户
String userName = request.getParameter("userName");
System.out.println("userName=="+userName);
//当前sessionId
//String sessionId = request.getSession().getId(); //删除当前sessionId绑定的用户,用户--HttpSession
//USER_SESSION.remove(SESSIONID_USER.remove(sessionId));
//删除当前登录用户已绑定的HttpSession
HttpSession session = USER_SESSION.remove(userName);//map中的remove方法返回删除value值 if (session != null){
//删除已登录的sessionId绑定的用户
SESSIONID_USER.remove(session.getId());
session.removeAttribute("userName");
session.setAttribute("userMsg", "您的账号已经在另一处登录,您被迫下线!");
}
} /**
*
* Description: 模拟DB登录判断<br>
* @param userName 用户
* @param password 密码
* @return
* @see
*/
private boolean login(String userName, String password){
return ("peizhongxian".equals(userName) && "123456".equals(password));
} /**
* 判断用户是否同时登陆同一个用户
*
* */
@RequestMapping(value="/checkUserOnline")
@ResponseBody
public void checkUserOnline(HttpServletRequest request,HttpServletResponse response) throws IOException{
HttpSession session=request.getSession();
PrintWriter out = response.getWriter();
out.print(session.getAttribute("userMsg"));
} }

  //监听器 MyListener

package com.zjn.oneLogin.dengLu;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener; public class MyListener implements HttpSessionListener { /**
* 实现HttpSessionListener接口监听 监听session的创建事件
*/
public void sessionCreated(HttpSessionEvent se){
String sessionId = se.getSession().getId();
System.out.println("创建session sessionId= " + sessionId);
} /**
* 实现HttpSessionListener接口监听 监听session的销毁事件
*/
public void sessionDestroyed(HttpSessionEvent se){
String sessionId = se.getSession().getId();
System.out.println("sessionId========="+sessionId);
//当前session销毁时删除当前session绑定的用户信息
//同时删除当前session绑定用户的HttpSession
LoginController.USER_SESSION.remove(LoginController.SESSIONID_USER.remove(sessionId)); System.out.println("销毁session sessionId= " + sessionId);
} }

  //web.xml 文件

 <listener>
<listener-class>com.zjn.oneLogin.dengLu.MyListener</listener-class>
</listener>

  

java保持同一时间同一账号只能在一处登录的更多相关文章

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

    一.问题引导 在Web开发中,实现一个账号只能在一处登陆有两种形式:1.当某个账号在某处登陆后,如果再在其他处登陆,将前一个账号挤掉:2.当某个账号登陆后,此账号在其他设备登陆提示已经登陆,无法登陆. ...

  2. asp.net mvc 简单实现一个账号只能在一个地方登录

    原理:  假设用户在机器A登陆后,  这时用户再次在机器B登陆,会以当前会话的SessionID作为键,用户id作为值,插入dictionary集合中,集合再保存在application(保存在服务器 ...

  3. shiro实现账号同一时间只能在一处登录(非单点登录)

    <bean id="myRealm" class="com.sys.shiro.MyRealm" /> <bean id="sess ...

  4. [Java]Java日期及时间库插件 -- Joda Time.

    来到新公司工作也有一个多月了, 陆陆续续做了一些简单的项目. 今天做一个新东西的时候发现了 Joda Time的这个东西, 因为以前用的都是JDK原生的时间处理API, 大家都知道Java原生的时间处 ...

  5. Java中日期时间API小结

    Java中为处理日期和时间提供了大量的API,确实有把一件简单的事情搞复杂的嫌疑,各种类:Date Time Timestamp Calendar...,但是如果能够看到时间处理的本质就可以轻松hol ...

  6. php 实现同一个账号同时只能一个人登录

    php 实现同一个账号同时只能一个人登录 张映 发表于 2015-01-22 分类目录: php 标签:mysql, nginx, openfire, php, redis 以前考虑过这个问题,今天实 ...

  7. (Java)微信之个人公众账号开发(二)——接收并处理用户消息(下)

    接下来,我们再讲一下图文消息: 如图: 大家可以先从开发者文档中了解一下图文消息的一些参数: 如上图,用户回复4时,ipastor返回了几条图文消息,上图中属于多图文消息,当然还有单图文消息,图文消息 ...

  8. Java 8 日期时间API

    Java 8一个新增的重要特性就是引入了新的时间和日期API,它们被包含在java.time包中.借助新的时间和日期API可以以更简洁的方法处理时间和日期; 在介绍本篇文章内容之前,我们先来讨论Jav ...

  9. java学习第13天( java获取当前时间,有关大数据的运算及精确数字运算,Date类)

    一 java获取当前时间 学习一个函数,得到当前时间的准确值 System.currectTimeMillis(). 可以得到以毫秒为单位的当前时间.它主要用于计算程序运行时间,long start= ...

随机推荐

  1. Centos7配置阿里epel源|yum源

    这一步非常重要.重要.重要.在这解释一下源的概念,打个比方如果手机想获取一个软件,可以选择很多途径,如华为的华为商店,小米的应用商店,苹果的App store,源就相当于各种手机获取软件的商店.因为国 ...

  2. 【Usaco 2009 Gold 】JZOJ2020年9月19日提高B组T2 电视游戏问题

    [Usaco 2009 Gold ]JZOJ2020年9月19日提高B组T2 电视游戏问题 题目 Description 农夫约翰的奶牛们游戏成瘾!本来FJ是想要按照陶叫兽的做法拿她们去电击戒瘾的,可 ...

  3. moviepy音视频剪辑:使用autoTrack、manual_tracking+headblur实现半自动追踪人脸打马赛克

    一.引言 在<moviepy1.03音视频剪辑:使用manual_tracking和headblur实现追踪人脸打马赛克>介绍了使用手动跟踪跟踪人脸移动轨迹和使用headblur对人脸进行 ...

  4. PyQt学习随笔:QtDesigner ListView控件列表项的初始化

    在QtDesigner中设计的界面中添加ListView控件后,是没办法添加需要在ListView控件中显示的列表项.由于ListView控件只是一个展示列表项的视图控件,实现了界面与数据的分离,其要 ...

  5. 从零开始的xxe学习

    本文介绍了一个菜鸡对xxe的一步步学习(内容多来源于大佬的博客,先感谢一波) 涉及知识点: (1)xxe 目录: 解析: 1.xxe是什么(不详解了,网上很多的) XXE(XML External E ...

  6. 写了一个类似与豆瓣的电影的flask小demo

    先展示页面 基本的功能是都已经实现了,更多那个地方是可以点的.只不过视频上面还用的宏,哎呀,感觉麻烦.有多麻烦呢,需要先定义一个宏,然后进行引用.我们才能是用,以我的观点,还不如直接是一个循环完事.. ...

  7. iOS崩溃日志 如何看

    日志主要分为六个部分:进程信息.基本信息.异常信息.线程回溯.线程状态和二进制映像. 我们在进行iPhone应用测试时必然会在"隐私"中找到不少应用的崩溃日志,但是不会阅读对于很多 ...

  8. js监测页面是否切换到后台

    最近做个弹幕,用的是第三方的插件,在浏览器页面切换到后台,返回后发现数据有堆叠卡死的情况,如何解决这个问题?网上参考了些demo,大致可以实现 1.document.hidden( Boolean值, ...

  9. vue项目中扫码枪收款

    扫码枪会将扫到的数据带入到获取焦点的输入框中,并且触发输入框的enter回车事件 1.页面上要有一个输入框,并且是获取焦点状态,当然它是隐藏的看不到,我是把宽高设置为0,然后加上回车事件.       ...

  10. XJOI contest 1590

    首先 热烈庆祝"CSP-S 2020全国开放赛前冲刺模拟训练题1"圆满结束!!! 感谢大毒瘤周指导的题目.题目还是很不错的,部分分设置的也比较合理,各种神仙随便 \(\text{A ...