使用 PHP 显示在线用户列表

Posted

技术标签:

【中文标题】使用 PHP 显示在线用户列表【英文标题】:Displaying a online user list using PHP 【发布时间】:2013-09-20 18:38:34 【问题描述】:

大家好,所以我有部分工作代码。但是我也有问题。但是让我开始告诉你我想要实现的目标......在我的网站上,我有一个登录工具,它会将所有用户发送到一个主页,并且只有当用户登录时。我想显示还有谁在我的页面上登录,在列表中显示用户名。

我的代码当前能力...

好的,所以我可以编写代码以将加入网站的用户添加到数据库中,然后将该数据库上的每个用户显示到一个 div 中,这非常棒,因为它可以在线显示每个人...

问题出在哪里...

但是,当用户离开页面时,需要从在线用户数据库中删除用户名。在过去,我会简单地使用 java 中的 on window close 选项来执行此操作。但由于 Google safari 和 Firefox 不再支持此选项,我不得不另谋出路。

下面的代码做了什么……

所以下面的代码会加载一个间隔,所以在很长一段时间后它会重复其中的代码。代码首先将用户添加到数据库中,然后将用户显示在站点上的列表中,然后从数据库中删除用户,以防万一他们离线。

问题出在哪里...

对于一个用户来说这一切都很好,但是一旦第二个用户参与其中,间隔就会在不同的时间下降。所以我在显示列表后从数据库中删除了用户,但这意味着第二个用户显示数据库,而第一个用户不在其中,因为他们被删除了......如果用户离线,这也会导致问题函数执行到一半时,它会将它们留在数据库中,并且不会将它们从数据库中删除。

以下是java代码...

    $(document).ready(function() 
   
var user_name = "<?php print $username ?>";

setInterval(function() 

       $.post('onlineusers.php',  name: user_name, action:"joined" ); 

       $.post("onlineusers.php",  action2: "list" , function(data)
       $('#listusers').html(data);
       );

       $.post("onlineusers.php",  nameleft: user_name, action3:"left" , function(data)
       $('#errorrreport').html(data);
       );

, 5000);
           
    ):  
    

现在有PHP文档代码

    //------------User joined page put into database --------  
if( $_REQUEST["name"])

  $user_name = $_REQUEST['name'];
;  

if( $_REQUEST["action"])

  $action = $_REQUEST['action'];  
;

if ($action == 'joined') 
  user_joined($user_name);   
;

//-----------Listing online users within page -------------  
 if( $_REQUEST["action2"])

  $action2 = $_REQUEST['action2'];

;

if($action2 == 'list') 

foreach (user_list() as $user)
       echo $user . "<br />";  
      
;
;

//------------ User left delete from the database ------------
if( $_REQUEST["nameleft"])

  $user_nameleft = $_REQUEST['nameleft'];

;  

if( $_REQUEST["action3"])
  $action3 = $_REQUEST['action3'];
;

if($action3 == 'left') 
  user_left($user_nameleft); 
;        



//------ Functions what to do... --------

    function user_joined($user_name) 
  $user_name = mysql_real_escape_string(htmlentities($user_name));
  mysql_query("INSERT INTO users (user_name)VALUES('$user_name')") or die("this didn't happen");
   


function user_left($user_nameleft) 
  $user_name = mysql_real_escape_string(($user_nameleft));
  $query = mysql_query("DELETE FROM users WHERE user_name = '$user_nameleft'")or die("failed to delete from table");



function user_list() 
  $user_list = array();
  $users_query = mysql_query("SELECT user_name FROM users") or die ("Unable to collect userlist");
  
while ($users_row = mysql_fetch_assoc($users_query))
  $user_list[] = $users_row['user_name'];

  
   
  return $user_list;


以上代码

抱歉,由于我多次重新编码以使其正常工作,因此有点混乱。 如果有人可以帮助我使其正常工作,我将不胜感激。现在,如果不仅仅是上面的代码,我可以在页面入口处将用户添加到在线用户数据库,并通过列出数据库中的用户来列出经常在线的用户。

真正的问题以及我在哪里寻求您的帮助...

但是,如果您对以下代码有任何想法,那就太好了...离开页面的用户是问题出现的地方。我需要一些方法来检查用户是否仍然处于活动状态,如果他们没有从数据库中删除它们,那么当列表刷新时,用户不再在列表中。例如 ping 某些东西直到它没有响应并从数据库中删除用户。

由于我当前的代码存在多个用户和间隔函数同步的问题,因此需要考虑到不同的用户会在不同的时间看到列表刷新。

附:我也看过使用 $SESSION 但是我仍然不确定如何通过检查离线用户然后从数据库中删除它们来完成这项工作,这可能是一种方法。

谢谢,我希望有足够的信息继续下去。

【问题讨论】:

我过去一直在玩这个,chrome/ff 很好地支持它,我认为它仍然支持,如果用户关闭选项卡它将他从列表中删除,它适用于 javascript 事件,但是opera没有内置这个事件处理程序,它计划了很多年,但是opera正在消亡,我认为它不会有朝一日。那么实现这一点的方法是将每个用户操作记录到数据库及其时间中,例如,如果时间超过 30 秒,请不要从在线用户列表中选择该用户。这在每个浏览器上都可以正常工作。 但是问题来了:如果它是一些聊天系统,而用户只是阅读消息而不执行任何操作,那么有很棒的 jQuery idle 插件可以跟踪用户鼠标事件:@ 987654321@ 真的很好用,我一直用它来显示聊天中的空闲/活跃用户。 【参考方案1】:

我采取了不同的方法: JS: 5 秒的循环,AJAX 请求 onlinenow.php 并在在线用户框中显示返回的 html - 这主要是从 W3C 复制和粘贴的,但 URL 已更改。

setInterval("getOnline()",5000);
function getOnline()

if (window.XMLHttpRequest)
  // code for IE7+, Firefox, Chrome, Opera, Safari
  xmlhttp=new XMLHttpRequest();
  
else
  // code for IE6, IE5
  xmlhttp=new ActiveXObject("Microsoft.XMLHTTP");
  
xmlhttp.onreadystatechange=function()
  
  if (xmlhttp.readyState==4 && xmlhttp.status==200)
    
    document.getElementById("onlineNow").innerHTML=xmlhttp.responseText;
    
  
xmlhttp.open("GET","http://www.path.com/to/onlinenow.php",true);
xmlhttp.send();

PHP: 获取用户/IP/标识符并在数据库中存储/更新 + unix 时间戳 返回过去 10 分钟内所有带有 unix 时间戳的行的列表

<?php
header('Access-Control-Allow-Origin: *');//Pretty sure this enables cross domain AJAX
$session = $_SERVER['REMOTE_ADDR'];
$time=time();
$time_check=$time-60; //SET TIME 10 Minute


$host="localhost"; // Host name
$username="root"; // Mysql username
$password="pass"; // Mysql password
$db_name="custom"; // Database name
$tbl_name="table"; // Table name


// Connect to server and select databse
 mysql_connect("$host", "$username", "$password")or die("cannot connect to server");
 mysql_select_db("$db_name")or die("cannot select DB");

// Check if user is already in the DB
$sql="SELECT * FROM $tbl_name WHERE session='$session'";
 $result=mysql_query($sql);

$count=mysql_num_rows($result);

if($count=="0")
    //User is not in the DB, lets add them
$sql1="INSERT INTO $tbl_name(session, time, ip)VALUES('$session', '$time', '".$_SERVER['REMOTE_ADDR']."')";
$result1=mysql_query($sql1);

//Update this user's entry
else 
    //User is in the DB, Update their entry
    $sql2="UPDATE $tbl_name SET time='$time' WHERE session = '$session'";
    $result2=mysql_query($sql2);


//Done updating info, time to get the user list
// if over 10 minute, delete session - Could just get all in past 10 mins instead
$sql4="DELETE FROM $tbl_name WHERE time<$time_check";
$result4=mysql_query($sql4);

//Get total users (Could get a list of names if you preferred and store the info)
$sql3="SELECT * FROM $tbl_name";
$result3=mysql_query($sql3);
$count_user_online=mysql_num_rows($result3);
if ($count_user_online == 1) 
    $plur = "";

else 
    $plur = "s";

echo "&nbsp;&nbsp;$count_user_online user".$plur." online now&nbsp;&nbsp;";

// Close connection 
mysql_close();
 ?>

我已经多次使用这个逻辑,有没有显示用户名(有时只是一个数字)。 代码当然可以改进,但基本原理是一个相当好的解决方案。

【讨论】:

是的,它很有意义并且可能很有用,请显示一些代码,以便我可以将它的一些逻辑用于将来使用,谢谢 :) 对不起,代码可能很乱,有一段时间没有做任何更改,但它可以工作! 谢谢,如果可以的话,斯蒂芬只是在搞乱一些混合。我并不是说这是答案,因为我希望看到一些不同的方法,因为看到其他人正在做什么以自己获得最佳方法总是很好的。【参考方案2】:

您可以尝试在插入时包含时间戳(基于服务器)而不是删除。

然后让 javascript 代码每分钟左右插入/更新一次。

那么您对谁在线的查询只是哪些行具有最近的时间戳(例如最后 65 秒)

当然,你最终必须清除表中的旧行,除非你想要记录每个人的日志,记录每个人的工作时间和时间。

【讨论】:

以上是关于使用 PHP 显示在线用户列表的主要内容,如果未能解决你的问题,请参考以下文章

c#如何实现显示在线用户列表的功能

我如何在 php、mysql 和 ajax 的会话中显示用户在线或离线

使用 Javascript 使用 ejabberd 获取在线用户列表?

XMPPFramework - 如何获取在线和离线好友列表?

在线时更新列表视图内容

使用 strophe 获取在线用户列表 xmpp [关闭]