- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我目前正在从事一个大型项目,该项目需要实现服务器发送的事件。我决定为此使用事件源传输,并从简单的聊天开始。目前客户端只监听一个新的聊天消息事件,但项目将来会有更多的事件。首先,我真的很关心服务器端脚本和其中的循环,其次,我不确定使用 mySQL 数据库作为存储(在本例中,用于聊天消息)是否可行实际上是一个很好的做法。当前循环在新消息出现在数据库中时泄露它们:
$statement = $connect->prepare("SELECT id, event, user, message FROM chat WHERE id > :last_event_id");
while(TRUE) {
try {
$statement->execute(array(':last_event_id' => $lastEventId));
$result = $statement->fetchAll();
foreach($result as $row) {
echo "id: " . $row['id'] . "\n";
echo "event: " . $row['event'] . "\n";
echo "data: |" . $row['user'] . "| >>> \n";
echo "data: " . $row['message'] . "\n\n";
$lastEventId++;
}
} catch(PDOException $PDOEX) {
echo $PDOEX->getMessage();
}
ob_flush();
flush();
usleep(10000);
}
据我所知,这样的循环是不可避免的,我的任务是优化它的性能。目前我在 while()
和 reasonable(?) usleep()
之外使用准备好的语句。
所以,对于那些有服务器端事件经验的人的问题:
感谢任何帮助,因为问题非常复杂,搜索信息不会给我任何提示或测试方法。
最佳答案
是否会同时连接所有 1000 多个用户?您是否将 Apache 与 PHP 一起使用?如果是这样,我认为您真正应该关心的是内存:每个用户都持有一个打开的套接字、一个 Apache 进程和一个 PHP 实例。您需要针对自己的设置自行衡量,但如果我们说每个 20MB,那么 1000 个用户需要 20GB 的内存。如果你收紧事情,那么每个进程都是 12MB,每 1000 个用户仍然是 12GB。 (一个 m2.xlarge EC2 实例有 17GB 的内存,所以如果你预算每 500-1000 个用户一个,我认为你会没问题。)
相比之下,对于 10 秒的轮询时间,CPU 使用率非常低。出于同样的原因,我不认为轮询 MySQL 数据库会成为瓶颈,但在这种使用级别上,我会考虑让每个数据库写入也对 memcached 进行写入。基本上,如果您不介意投入一些硬件,您的方法看起来是可行的。它不是最有效的内存使用方式,但如果您熟悉 PHP,它可能是最有效地使用程序员时间的方式。
更新:刚看到 OP 的评论,意识到 usleep(10000)
是 0.01s,而不是 10s。哎呀!这改变了一切:
我会使用队列服务而不是 memcached,您可以找到现成的东西,或者很容易地用 PHP 编写自定义的东西。您仍然可以将 MySQL 作为主数据库并让您的队列服务轮询 MySQL;这里的区别是你只有一个进程集中轮询它,而不是一千个。队列服务是一个简单的套接字服务器,它接受来自每个前端 PHP 脚本的连接。每次轮询发现一条新消息时,它都会将其广播给所有连接到它的客户端。 (有不同的方法来构建它,但我希望这能给你一个大概的想法。)
在面向前端的 PHP 脚本中,您使用了具有 15 秒超时的 socket_select()
调用。它仅在没有数据时唤醒,其余时间使用零 CPU。 (15 秒超时是为了让您可以发送 SSE keep-alives。)
关于php - 事件源的性能,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20471034/
语境 我正在将一个旧的 php 电子商务网站变成一个用 gatsby.js 构建的静态网站。 我将所有产品元数据放入单独的 .json 文件(每个产品一个),并且我能够使用 json 和文件转换器插件
我曾经能够使用三指向上滚动在源/标题之间切换。自从升级到 Lion 后,我只进行常规滚动。有没有办法恢复该功能? Aka,当我像以前那样向上/向下滚动时,它不会跳到对应的位置。 更新 Apple 在
我有一个包含复选框输入的克隆元素。当克隆的元素未被选中时,我需要也取消选中源元素。有没有办法在 jQuery 中做到这一点?或者我是否以错误的方式处理这个问题(即使用clone())?我应该提到我的问
我有一个类,其中有两个 JSpinner 对象,x 和 y。我有一个更改监听器,它已添加到两者中。有人可以告诉我如何实现我的更改监听器,以便监听器可以区分两个对象之间的区别。例如伪代码: if(sou
我正在编写涉及 for 循环的代码,该循环在每个索引处进行计算。 这些计算中的最小值存储在一个变量中,我在程序末尾使用 MPI_Allreduce 来确定所有进程的全局最小值。 但是,我需要一种方法来
我需要在一个 Android 项目中创建一个 map View ,我从服务器获取自定义 map 图 block PNG。有人知道实现此类功能的简单许可 API 吗? 最佳答案 我使用了 OsmDroi
因为我必须创建一个可以更改图像 (src/background-url) 的函数。我想知道如何识别标签以及它是使用 src 还是 url 来访问图像。 让我们说 早些时候我写了一个可以
当我使用源 map 浏览器 https://github.com/danvk/source-map-explorer要检查捆绑包中的内容,我得到以下输出: D:\projects\angular\mT
我正在为客户将 Windev 应用程序移植到 Objective-C。出于显而易见的原因,使用以前的源代码会更简单。 不幸的是,它是加密的,我需要 EDI 才能看到它;完整版的 Windev 太贵了(
我有一个简单的视频播放器,它使用 WPF MediaElement 播放一系列视频。这些视频一起形成一个围绕静止图像移动的连续电影。在每个视频结束时,运动会卡住在当前播放视频的最后一帧。当我按下一个按
我需要更改 openlayer 的图层源(使用 open weather api)。目前我正在使用以下代码但没有成功。 let layer = this.map.getLayers().getArra
我正在尝试在 /dev/random 的机器上运行代码不会很快填满,我正在尝试使用的 Java 程序因缺少随机数而挂起。/dev/urandom产生“不太好”的随机数,但不会阻塞,对于这种情况,我宁愿
我需要 Yocto 项目的源代码包。我已经拥有整个项目的所有资源,但它们还包括开发工具。 我想有一种方法来生成将为目标图像构建的所有包的(修补的)源。因此,例如,如果目标图像包含 busybox,我想
如何对入侵者隐藏 iFrame src 假设我正在流式传输我的网络摄像头或我的电脑屏幕,这是 iframe 代码: 并且我不希望它在大多数浏览器上显示页面源中的流 URL 和检查功能! 这意
是否可以进入 Qt 源,例如qmainwindow.cpp,在 Qt Creator 中?目前我正在看到反汇编,但最好能看到源代码。 最佳答案 当然!但您可能必须首先: 转到 $QT_HOME/qt
我正在尝试创建一个包含很少动漫剧集的简单网站。我有一个关于 javascript 的问题。如何通过单击我的链接之一来更改视频源?我明白,我必须使用事件监听器,只需更改 取决于我点击的链接,但我不太擅长
我有一个带有 BindingSouce 的 DevExpress GridControl。我想清除 BindingSource 并用新数据填充它。我这样做: var list = new List()
当单击提交输入按钮时,我尝试将其他参数(选定复选框的列表)传递到服务器处理的 DataTables 表#my_table: 这可能意味着我必须将 my_table.sAjaxSource 设置为后端脚
(好吧,别对我大喊大叫,这里已经很晚了:)) 我正在研究 delta diff 工具(命令行工具或组件,只要我可以从 Delphi 2010 调用它们就可以了) 我有这个项目,我将文件上传到服务器,我
我需要解析 Yahoo Weather RSS feed 中的某个位置,例如 http://weather.yahooapis.com/forecastrss?w=44418&u=c例如,获取最高、最
我是一名优秀的程序员,十分优秀!