我正在尝试在网上制作一个简单的聊天应用,使用长轮询& PHP。
主要是我在客户端有2个ajax功能,
1.更新:更新聊天消息
2.发送:发送用户输入的消息
更新功能执行长轮询,即等待10秒或直到新消息可用。
send函数写入数据库。
我的问题是,当更新功能正在运行(长轮询)时,send函数无法运行。只有在更新功能运行完毕后,才会运行send函数。这是预期的行为还是我的代码有问题?
如果你想看到网络,你可以在这里访问它:http://tedhost.awardspace.us
如果你懒得注册,你可以使用用户名dummy0和密码123456
这是代码。如果我的代码很乱,请谨慎。
ajax代码:http://tedhost.awardspace.us/ajax.js
update_chat.php
<?php
session_start();
$filename = "wew.xt";
$user = $_SESSION["user"];
$last = $_SESSION["lmsgtime"];
if (file_exists($filename)) {
$lama = 0;
$mulai = time();
do {
$fin = fopen($filename, "r");
$current = 0;
fscanf($fin, "%d", $current);
fclose($fin);
usleep(10000);
$akhir = time();
} while($current <= $last && $akhir - $mulai <= 10);
}
include "con.php";
mysql_select_db($dbname, $con);
$sql = "SELECT * FROM chats WHERE time > '$last' ORDER BY time ASC";
$res = mysql_query($sql);
/*
$lama = 0;
while (mysql_num_rows($res) <= 0 && $lama <= 10000000) {
$res = mysql_query($sql);
usleep(500);
$lama += 500;
}
*/
if (!$res)
die("error");
$out = "";
while($row = mysql_fetch_array($res)) {
$out = $out . "<div id=";
if ($row["user"] == $user)
$out = $out . "\"chatme\"";
else if ($row["user"] == "sys")
$out = $out . "\"chatsystem\"";
else
$out = $out . "\"chatother\"";
$out = $out . ">";
$out = $out . date("(h:i:s) ", $row["time"]);
$out = $out . $row["user"];
$out = $out . ": " . stripslashes($row["data"]);
$out = $out . "</div>";
$last = $row["time"];
}
$_SESSION["lmsgtime"] = $last;
echo $out;
?>
send.php
<?php
$time = time();
session_start();
include "con.php";
$data = mysql_real_escape_string($_GET["msg"]);
$user = mysql_real_escape_string($_SESSION["user"]);
mysql_select_db($dbname, $con);
$sql = "INSERT INTO chats VALUES ('$time', '$user', '$data')";
$res = mysql_query($sql);
if (!$res)
die("err");
else
echo "ok";
$lho = fopen("wew.xt", "w");
fprintf($lho, "%d\n", $time);
fclose($lho);
?>
答案 0 :(得分:2)
如果您正在使用默认会话处理程序,那么只有一个脚本可以一次打开会话,因为一次只能有一件事写入会话文件,否则可能会发生损坏。
您可以通过在脚本中尽快调用session_write_close来解决此问题。对于只读取会话数据的脚本,您应该能够在session_start之后直接调用它。对于对会话进行更改的脚本,您必须在最后一次更改会话后立即调用它。
虽然使用session_write_close意味着您不会保存对$ _SESSION所做的任何后续更改,但$ _SESSION数组仍然可以访问并包含最后从会话加载的值。