一旦你在 ajax 请求上使用套接字,你应该保持你的 Phalcon 保持原样,并尝试使用 node.js+socket.io 实现简单的工具。
最简单的方法是在 Node 中创建一个事件转发器,它会在一侧监听事件,并将它们转发到用户浏览器。更多描述here。
在您的情况下,我建议在您的 Phalcon 模型中添加一个 afterSave 监听器,如 documentation 中所述。在afterSave 方法期间,您将能够通过例如。向您的 Node 服务发送一个 UDP 数据包,其中包含 X 表中有新记录的信息。
您的 Node 服务应该将此事件转发给客户端的浏览器,由 javascript 决定当前用户是否正在查看 X 表。如果是,它应该锁定视图以防止任何操作并向 Phalcon 服务发起 ajax 请求,该服务应该向您的数据库询问实际数据并刷新视图中的 HTML 内容。
当然,您可以创建一个 Node 服务,它可以直接在 DB 上侦听更改,并将适当的事件发送给适当的用户,所有内容都准备好完全动态更新。但这是一种先进的方法,可能有点矫枉过正。我想在这里强调一下,只要您的 Phalcon 服务针对速度进行了优化并且发送的数据不是太大,建议的解决方案就可以在几分之一秒内工作,给人一种实时的感觉。
当然,您可以使用 Phalcon 而不是 Node 创建转发服务,但时间长了您会后悔的,因为使用基于事件的工具来维护这种基于事件的脚本更容易。同样对于不时使用 JavaScript 的高级 PHP 程序员来说,快速学习这么一点点 Note.js 以做出如此简单的解决方案甚至都不是挑战。
使用 PHP 通过套接字发送包
$host = array(
'scheme' => 'udp', // udp makes it lightweight and connectionless
'host' => '192.168.10.10' // choose an IP where node is running
'port' => '8888' // choose one > 1023
);
$param = sprintf('%s://%s:%s', $host['scheme'], $host['host'], $host['port']);
$socket = fsockopen($param, $errno, $errstr, $timeout);
我确实在Socket 类中包含了类似的内容。我宁愿使用持久套接字(pfsockopen),因此即使看起来不是这样,一次很少有进程会使用这个套接字。在afterSave 我使用Send 方法,或多或少:
fwrite($socket, json_encode($msg));
基于 websocket 的 Node.js 事件转发脚本
示例配置文件node.json:
{
"wsServer": {
"listeners": {
"udp": {
"port": 8888
}
},
"server": {
"port": 8000
}
}
}
您的依赖项将是 socket.io 和 dgram。一旦你学会了如何创建 Node 应用程序,你稍后就会明白我的意思。
为了便于理解以下内容:
var config = require('node.json').wsServer;
var app = require('http').createServer().listen(config.server.port);
var io = require('socket.io').listen(app);
var listenerDgram = require('dgram').createSocket('udp4');
listenerDgram.bind(config.listeners.udp.port);
var users = {};
io.sockets.on('connection', function(socket) {
// if you make user to connect by his individual ID during
// websocket connection, providen after ?, like ?1234
var user = parseInt(socket.handshake.query.user);
// here to save user into var users
if (!users[user]) {
users[user] = {
sockets: [socket]
};
} else {
users[user].sockets.push(socket);
}
socket.on('disconnect', function() {
// removing user from var users
// warning: socket by socket, and if last
// socket is closed, remove whole user section
});
});
// to emit data to all sockets of choosen users
emit = function(sockets, message, data) {
for (var x in sockets) {
if (sockets[x]) {
sockets[x].emit(message, data);
}
}
};
// now UDP listening section
listenerDgram.on('message', function(msg, rinfo) {
// you can declare checkIncoming if you have standarized
// frames, or just use msg as it is
var _dat = checkIncoming(msg.toString().trim(), true);
var _response = {
action: _dat.action,
data: _dat.data
};
// you can declare standarizeFrame to define your own protocol,
// or just use _response.
var frame = standarizeFrame(_response);
if (_dat.user) { // emitting to single user declared
emit(users[_dat.user].sockets, 'notification', frame);
} else { // emitting to all connected users
io.emit('notification', frame);
}
});
因此,您将 JSON 化的字符串发送到 Node 在端口 8888 上运行的服务器,并且用户 Web 界面连接到端口 8000 上的同一主机以接收它们。大量的调试工作摆在您面前。
PS:很抱歉这么武断的回答,但我自己走过它,因为我爱上了 Phalcon 并且不知道 Node.js 是否存在。给自己时间学习 Node 帮助我将基于事件的解决方案减少到绝对最低限度,将它们包含在 100 行而不是数千行中,使它们易于维护,让我的服务器深呼吸 CPU 和比基于 php 的客户端更满意.