ASP.NET实现唯一登陆


一(踢人):

简单,我做过类似的,数据库里加一个字段,Guid,然后用户登陆时,创建一个唯一字段,把这个字段存到数据库中和COOKIE中,然后用户访问时再判 断,COOKIE中的字段是否与数据库中的一致,如果不一致,就是帐号复复登陆了(因为另一个帐号登陆时,会重新修改GUID字段,所以会造成与本地不一 致),这时就提示他,“您的帐号在其它地方登陆”,这样就是同一个IP也不能同时重复登陆

 

二:

  如果你随便写写的话 就在数据库加上个字段就行了,退出的时候设置为0,
但是往往很多中情况都会发生,比如当电脑突然断电,或者突然死机,或者电脑突然爆炸,这些都是不能更改数据库的那字段

我有两个思路,1个就是 session+global全局配置文件(就是添加文件--后缀名.asax那个),
session 有个生命周期可以随意设置,而且global全局配置文件里面还有个session消失的事件(Session 必须保存在进程内即Inproc)
那就好办了,直接写就行,不怕电脑炸了,


  2:运用ajax+settiomout时时更新
后台写个共ajax调用的方法,功能是向数据库添加当前时间精确到秒
前台的settime()方法每十秒或者更短时间 执行此ajax

当然第一次加载的时候是在body里面的 onload事件中先执行的 

  在web开发时,有的系统要求同一个用户在同一时间只能登录一次,也就是如果一个用户已经登录了,在退出之前如果再次登录的话需要报错。

  常见的处理方法是,在用户登录时,判断此用户是否已经在Application中存在,如果存在就报错,不存在的话就加到 Application中(Application是所有Session共有的,整个web应用程序唯一的一个对象):

C#代码

  1. string strUserId = txtUser.Text;   
  2.   ArrayList list = Application.Get("GLOBAL_USER_LIST") as ArrayList;   
  3.  if (list == null)   
  4.   {   
  5.  list = new ArrayList();   
  6.   }   
  7.  for (int i = 0; i < list.Count; i++)   
  8.   {   
  9.  if (strUserId == (list[i] as string))   

10.   {   

11.  //已经登录了,提示错误信息   

12.   lblError.Text = "此用户已经登录";   

13.  return;   

14.   }   

15.  }   

16.   list.Add(strUserId);   

17.  Application.Add("GLOBAL_USER_LIST", list);  

 string strUserId = txtUser.Text;

  ArrayList list = Application.Get("GLOBAL_USER_LIST") as ArrayList;

  if (list == null)

  {

  list = new ArrayList();

  }

  for (int i = 0; i < list.Count; i++)

  {

  if (strUserId == (list[i] as string))

  {

  //已经登录了,提示错误信息

  lblError.Text = "此用户已经登录";

  return;

  }

  }

  list.Add(strUserId);

  Application.Add("GLOBAL_USER_LIST", list);

 当然这里使用Cache等保存也可以。

  接下来就是要在用户退出的时候将此用户从Application中去除,我们可以在Global.asax的Session_End事件中处理:

C#代码

  1. void Session_End(object sender, EventArgs e)   
  2.   {   
  3.   // 在会话结束时运行的代码。   
  4.   // 注意: 只有在 Web.config 文件中的 sessionstate 模式设置为   
  5.   // InProc 时,才会引发 Session_End 事件。如果会话模式设置为 StateServer   
  6.   // 或 SQLServer,则不会引发该事件。   
  7.   string strUserId = Session["SESSION_USER"] as string;   
  8.   ArrayList list = Application.Get("GLOBAL_USER_LIST") as ArrayList;   
  9.   if (strUserId != null && list != null)   

10.   {   

11.   list.Remove(strUserId);   

12.   Application.Add("GLOBAL_USER_LIST", list);   

13.   }   

14.   }  

void Session_End(object sender, EventArgs e)

  {

  // 在会话结束时运行的代码。

  // 注意: 只有在 Web.config 文件中的 sessionstate 模式设置为

  // InProc 时,才会引发 Session_End 事件。如果会话模式设置为 StateServer

  // 或 SQLServer,则不会引发该事件。

  string strUserId = Session["SESSION_USER"] as string;

  ArrayList list = Application.Get("GLOBAL_USER_LIST") as ArrayList;

  if (strUserId != null && list != null)

  {

  list.Remove(strUserId);

  Application.Add("GLOBAL_USER_LIST", list);

  }

  }

 这些都没有问题,有问题的就是当用户直接点浏览器右上角的关闭按钮时就有问题了。因为直接关闭的话,并不会立即触发Session过期事件,也就是关闭浏览器后再来登录就登不进去了。

  这里有两种处理方式:

  1、使用Javascript方式

  在每一个页面中加入一段javascript代码:

Js代码

  1. function window.onbeforeunload()   
  2.   {   
  3.  if (event.clientX>document.body.clientWidth && event.clientY<0||event.altKey){   
  4.   window.open("logout.aspx");   
  5.   }   
  6.  }  

 function window.onbeforeunload()

  {

  if (event.clientX>document.body.clientWidth && event.clientY<0||event.altKey){

  window.open("logout.aspx");

  }

  }

 由于onbeforeunload方法在浏览器关闭、刷新、页面调转等情况下都会被执行,所以需要判断是点击了关闭按钮或是按下Alt+F4时才执行真正的关闭操作。

  然后在logout.aspx的Page_Load中写和Session_End相同的方法,同时在logout.aspx中加入事件:onload="javascript:window.close()"

  但是这样还是有问题,javascript在不同的浏览器中可能有不同的行为,还有就是当通过文件->关闭时没有判断到。

  2、使用xmlhttp方法(这种方法测试下来没有问题)

  在每个页面中加入如下的javascript(这些javascript也可以写在共通里,每个页面引入就可以了)

Js代码

  1. var x=0;   
  2.  function myRefresh()   
  3.   {   
  4.  var httpRequest = new ActiveXObject("microsoft.xmlhttp");   
  5.   httpRequest.open("GET", "test.aspx", false);   
  6.   httpRequest.send(null);   
  7.   x++;   
  8.  if(x<60) //60次,也就是Session真正的过期时间是30分钟   
  9.   {   

10.  setTimeout("myRefresh()",30*1000); //30秒   

11.  }   

12.   }   

13.  myRefresh();  

 var x=0;

  function myRefresh()

  {

  var httpRequest = new ActiveXObject("microsoft.xmlhttp");

  httpRequest.open("GET", "test.aspx", false);

  httpRequest.send(null);

  x++;

  if(x<60) //60次,也就是Session真正的过期时间是30分钟

  {

  setTimeout("myRefresh()",30*1000); //30秒

  }

  }

  myRefresh();

在web.config中设置

Xml代码

  1. <sessionState mode="InProc" timeout="1"></sessionState>   

<sessionState mode="InProc" timeout="1"></sessionState>

  test.aspx页面就是一个空页面,只不过需要在Page_Load中加入:

C#代码

  1. Response.Expires = -1;  

 Response.Expires = -1;

 保证不使用缓存,每次都能调用到这个页面。

  原理就是:设置Session的过期时间是一分钟,然后在每个页面上定时每30秒连接一次测试页面,保持Session有效,总共连60次,也就是30分钟。如果30分钟后用户还没有操作,Session就会过期。当然,如果用户直接关闭浏览器,那么一分钟后Session也会过期。这样就可以满足要求了

你可能感兴趣的:(asp.net)