基于PHP的聊天室

基于PHP的聊天室

从前有一个相当流行的web聊天室,叫做Star Trekker 聊天室。我能够来到这个聊天室要感谢我的一个朋友,并且即使Star Trekker的聊友们几乎与我不是同一个圈中的,但是我发现对于他们的大部分人来说,都很友好和有意思。但是当Star Trekker关闭之后,这不得不谢谢运行在后台的Perl了,是它吞掉了服务器的资源,所以这些快乐和友善的人们不得不离开,无处可去了。我有幸在那段时间开张了自已的仿Star

Trekker的聊天室,并且开始设法与许多原来的Trekker的那些无家可归的聊友进行联系。出于对Perl所引起的资源消耗问题的警惕,当一个朋友向我推荐PHP时,我非常的高兴。

这个经过精心设计的web聊天室使用了从表单传递过来的变量,并且把它们处理为HTML ,然后将其写入文件。把表单和信息文件放在一个框架中,你可以看到它与一个叫BeSeen的聊天室很象。当然它的好处在于,

我们的聊天室比起它的BeSeen堂兄弟来说要聪明一些。

< form action="chat.php3" method="post" >

Name : < input type="text" name="name" >< br >

Message : < input type="text" name="message" >< br >

< input type="submit" value="Send" >

< /form >

上面就是基本的用于输入的表单。你可能想把它弄得更美观一些,但是不管出于什么目的,这个就是你要处理的。它发送两个变量给chat.php3,分别叫做$name和$message。

不过,在处理那些变量之前,我们需要从消息文件中把当前的内容取出来,否则在一个时间里我们只能看到一条消息了。几乎没有一个方法可以管理对话。只有象我一样对自已的消息文件的结构很熟悉,我就知道每一条消息都以一个回车符结束。这就是说,可以使用file()函数来将消息文件读到一个数组中去。

消息文件有12行。在12行中,第1行为头信息,第2行到11行为旧的消息,第12行包含了我的脚标。

我最感兴趣的是得到一个能够包含所有那些旧消息的字符串。

< ?php

// 把文件读到一个数组中

$message_array = file("messages.html");

// 编辑字符串

for ($counter = 1; $counter < 10; $counter++) {

$old_messages .= $message_array[$counter];

}

? >

在处理字符串的时候,我将for循环的$counter初始化为1而不是0。这是因为我知道$message_array 数组的第0个元素包含的是我的头信息,我不需要它。而且,通过设置循环结束条件为$counter < 10,意思是只有数组中1到9的元素被读到字符串中。对于剩下的两个元素,第11个包含我的脚标,第10个包含的是最旧的消息。这两个我都想删除,因为在任何时刻我只让屏幕显示10条消息。修改$counter < 10 表达式,可以允许你改变所包含消息的数量。

现在已经有了旧的消息,接着我想生成新的消息。我们已经有了两个变量:$name 和$message,所以写出一个新的消息就很容易了。

< ?php $new_message = "$name : $message< br > "; ? >

我们就快要写好消息文件了。剩下需要的就是头信息和脚标。先加入简单的头部信息:

< ?php

// 除了在字符串末尾有回车符,其它地方不能有回车符,这一点很重要。

// 要把所有的头信息放在一样。

$header = "< html >< body bgcolor="#000000" text="#ffffff" > ";

? >

我们想让消息屏幕能够自动刷新,那么人们就可以看到新的贴子了。我没有使用javascript,而是采用了META标记进行刷新,主要是因为它可能更容易被客户端支持。我也不想让搜索引擎对我的消息文件进行索引。所以重新定义头信息为:

< ?php

$header = "< html >< head >< meta http-equiv="refresh" content="8" >".

"< meta name="robots" content="noindex" >< /head >".

"< body bgcolor="#000000" text="#ffffff" > ";

? >

在文件的脚标处,我一般是放置一小段版权信息,还有与打开的头信息对应的结束标记。

< ?php

$footer = "< p align="center" >< font color="#000000" >".

"© Mike Hall 2000< /font >< /p >< /body >< /html >";

? >

将版权信息用< font color="#000000" >包起来表示只有被选中它才会被看见,因为它的颜色与背景色#000000一样。这样做的目的只是为了不让它影响显示。

现在我们终于有了写新文件所需的所有东西了:

< ?php

// 打开文件,并且将文件长度截为0

$open_file = fopen("messages.html", "w");

// 写入文件的头信息

fputs($open_file, $header);

// 新的一行

// (使用stripSlashes,因为我们不想让所有的转义字符出现在消息文件中)

fputs($open_file, stripslashes($new_message));

// 旧的行

fputs($open_file, $old_messages);

// 脚标

fputs($open_file, $footer);

// 关闭文件

fclose($open_file);

? >

现在我们有了需要通过$REMOTE_ADDR变量来交叉引用的文件,这样我们可以区分出想要发贴的用户是否已经被扁或没有被扁。很简单:

< ?php

for ($counter=0;$counter< sizeof($banned_array);$counter++) {

if ($banned_array[$counter] == $REMOTE_ADDR) {

print("< font color="red" align="center" >".

"You have been banned from this chat< /font >");

exit;

}

}

? >

exit命令将立即停止脚本的执行。在开始对传递过来的变量执行处理之前,插入对被扁用户的检查,这样被扁用户就不能使用聊天室了。

比较好的解决在某些情况下动态IP地址的问题的一个意见就是,检查IP地址块的所属范围。一个简单的函数可以容易地实现它。

< ?php

function makeMask($ip) {

// remember to escape the . so PHP doesn’t think it’s a concatenation

$ip_array = explode(".", $ip);

$ip_mask = "$ip_array[0].$ip_array[1].$ip_array[2]";

return $ip_mask;

}

? >

然后我们把循环中的if替换成

< ?php

for ($counter=0;$counter< sizeof($banned_array);$counter++) {

if (makeMask($REMOTE_ADDR) == makeMask($banned_array[$counter])) {

print("< font color="red" align="center" >".

"You have been banned from this chat< /font >");

exit;

}

}

? >

我们有了针对动态IP地址的保护措施。

最后我们需要一种方法最先得到惹麻烦的IP。我的实现是将$name和$REMOTE_ADDR记录到一个名为

iplist.html的文件中。对于一个分离的,秘密的URL,我可以在浏览消息的同时监控IP地址。这可以增加一些意外的好处,就是能够发现假冒者--在这些地方最常犯的“罪”。

iplist.html与messages.html的创建方法基本上一样。首先将当前的值从iplist.html中取出来,我们剥离掉头信息,脚标和旧的IP记录,然后创建一个新的记录,新的头信息,新的脚标。为了让布局更清楚,我使用了表格。

< ?php

$header = "< html >< body bgcolor="#000000" text="#ffffff" >< table > ";

$footer = "< /table >< /body >< /html >";

$new_ip = "< tr >< td >$name< /td >< td >$REMOTE_ADDR< /td >< /tr > ";

$ip_array = file("iplist.html");

for ($counter = 1; $counter < 20; $counter++)

$old_ips.= $ip_array[$counter];

? >

简单地把内容写入磁盘与对message文件所做的一样,这样我们就有了一个web聊天室。比Java有更好的跨越平台的兼容性,并且除了一个web浏览器什么都不需要了--我听说甚至Dreamcast就是这样工作的!

有一些东西你可能想试着自已做一下,包括合并一些常用代码片段为函数,编写一个可以自动增加惹麻烦的人到被扁列表中的脚本和编写一个regex表达式,可以扫描消息正文中的URL和e-mail,并将之自动转换成链接(象Outlook Express 和ICQ做的那样)。

试一试,体会一下乐趣,得到一些经验。这就是在PHP方面我是如何开始的并且现在我已经以它为职业了。

祝聊天快乐!