常有碰到有同学问在线人数的问题,官方默认的是显示最大在线500人 (建议不要改动太大的值,影响页面加载的效率) 。这里我就和大家一起看看,在线人数是如何设置和现实的。
首先我们找到论坛首页网站模板的处理程序页面,source/module/forum/forum_index.php 找到如下代码:

  1. $onlineinfo = explode(" ", $_G['cache']['onlinerecord']);
  2. if(empty($_G['cookie']['onlineusernum'])) {
  3.                         $onlinenum = DB::result_first("SELECT count(*) FROM ".DB::table('common_session'));
  4.                         if($onlinenum > $onlineinfo[0]) {
  5.                                 $onlinerecord = "$onlinenum ".TIMESTAMP;
  6.                                 DB::query("UPDATE
    ".DB::table('common_setting')." SET svalue='$onlinerecord' WHERE
    skey='onlinerecord'");
  7.                                 save_syscache('onlinerecord', $onlinerecord);
  8.                                 $onlineinfo = array($onlinenum, TIMESTAMP);
  9.                         }
  10.                         dsetcookie('onlineusernum', intval($onlinenum), 300);
  11.                 } else {
  12.                         $onlinenum = intval($_G['cookie']['onlineusernum']);
  13.                 }

这里$onlineinfo 为在线信息的缓存记录,如果此值为空,则查询 common_session 表,读取在线人数的缓存。如果当前人数大于缓存最高在线人数,则更新 common_setting 表的最高在线记录人数,并缓存。如果有在线人数记录则直接读取在线人数。

如果后台开启了现实在线人数,则处理在线现会员信息。

  1. $_G['setting']['maxonlinelist'] = $_G['setting']['maxonlinelist'] ? $_G['setting']['maxonlinelist'] : 500;

在此程序对默认显示的在线人数默认限定为500

  1. $query = DB::query("SELECT uid, username, groupid, invisible, lastactivity, fid FROM ".DB::table('common_session')." WHERE uid>'0' LIMIT ".$_G['setting']['maxonlinelist']);
  2.                         while($online = DB::fetch($query)) {
  3.                                 $membercount ++;
  4.                                 if($online['invisible']) {
  5.                                         $invisiblecount++;
  6.                                         continue;
  7.                                 } else {
  8.                                         $online['icon']
    = !empty($_G['cache']['onlinelist'][$online['groupid']]) ?
    $_G['cache']['onlinelist'][$online['groupid']] :
    $_G['cache']['onlinelist'][0];
  9.                                 }
  10.                                 $online['lastactivity'] = dgmdate($online['lastactivity'], 't');
  11.                                 $whosonline[] = $online;
  12.                         }

这里通过 common_session 表的查询,搜索出最近在线的显示会员,这里排除了 invisible 的隐身会员。

  1. if(isset($_G['cache']['onlinelist'][7]) && $_G['setting']['maxonlinelist'] > $membercount) {
  2.                                 $query = DB::query("SELECT uid,
    username, groupid, invisible, lastactivity, fid FROM
    ".DB::table('common_session')." WHERE uid='0' ORDER BY uid DESC LIMIT
    ".($_G['setting']['maxonlinelist'] - $membercount));
  3.                                 while($online = DB::fetch($query)) {
  4.                                         $online['icon'] = $_G['cache']['onlinelist'][7];
  5.                                         $online['username'] = $_G['cache']['onlinelist']['guest'];
  6.                                         $online['lastactivity'] = dgmdate($online['lastactivity'], 't');
  7.                                         $whosonline[] = $online;
  8.                                 }
  9.                         }

这里如果在最大在线人数大于当前在线会员,这剩余的部分通过游客补全。

  1. if($onlinenum > $_G['setting']['maxonlinelist']) {
  2.                                 $membercount = $discuz->session->onlinecount(1);
  3.                                 $invisiblecount =
    DB::result_first("SELECT COUNT(*) FROM ".DB::table('common_session')."
    WHERE invisible = '1'");
  4.                         }
  5.                         if($onlinenum < $membercount) {
  6.                                 $onlinenum = $discuz->session->onlinecount(0);
  7.                                 dsetcookie('onlineusernum', intval($onlinenum), 300);
  8.                         }
  9.                         $guestcount = $onlinenum - $membercount;

以上代码的意思是,如果在线人数> 最大显示在线人数,则重新统计在线会员人数。

一旦在线人数< 在线会员数,则重新统计所有在线会员数 (包括隐身) 并写入缓存。

最后由在线总人数和在线会员数得出在线游客。