- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我将尝试解释我的实际设置,其背后的想法,出现的问题以及围绕它的尝试。
上下文
我有一个PHP5.3后端向Javascript(使用jQuery 1.7.x)提供“事件”(事件是一个包含一些数据的标准数组,其中包括唯一的序列号)。使用jsonp(在子域上)并在服务器端进行长轮询来检索事件。第一个事件的ID为1,然后随着每个新事件的增加而递增。客户端跟踪“上次检索到的事件ID”,该值从0开始。对于每个长轮询请求,它都提供该ID,因此后端仅返回在该事件之后发生的事件。
按照以下方式处理事件:接收到事件(通过jsonp回调)后,将它们存储在eventQueue变量中,并将“最后检索到的事件ID”更新为接收到并存储在队列中的最后一个事件。然后调用一个函数来处理下一个排队的事件。该函数检查事件是否已经被处理(通过在事件开始处理时设置的另一个变量的方式),如果不执行任何操作,则调用堆栈将我们带回到jsonp回调,其中新发出长轮询请求。 (这将在处理其他事件时重复排队新事件的过程。)但是,如果当前没有正在处理的事件,它将验证队列中是否还有事件,如果是,它将处理第一个事件(具有最低ID)。 “处理事件”可以是与我的应用程序相关的各种任务,但与我所遇到的问题或上下文无关。例如,更新变量,页面上的消息等。将某个事件视为“正在处理”(某些事件会进行ajax调用以获取或发送数据,在这种情况下,此操作会在其成功的ajax回调中发生),调用另一个名为eventComplete的函数。该函数从事件队列中删除已处理的事件,确保将用于处理是否正在处理事件的变量设置为false,然后调用处理事件队列的函数。 (因此它处理下一个最低的ID事件)
问题
这在所有经过测试的主要浏览器上也都非常有效。 (在Internet Explorer 8和9,Chrome,Opera和Firefox上进行了测试)由于使用了长时间轮询,因此它也非常灵活。获得所有已发生的事件并保持应用程序完全相同的状态的所有“历史记录”(大多数事件会生成文本数据,并将其附加到页面的某种控制台中)也非常好,即使在重新加载页面之后也是如此。然而,当事件数量变多时,这也成为问题。根据估计,我将需要能够处理多达30,000个事件。在我的测试中,即使在7,000个事件中,情况也开始变差。 Internet Explorer 8堆栈溢出约400个事件。 Chrome不会加载所有事件,但是会关闭(并且中断,但并不总是在同一点,这与IE8不同)。 IE9和FF处理得很好,并且在处理所有事件时挂起2-3秒,这是可以容忍的。但是,我认为这可能只是更多事件破裂之前的问题。我是不是对当前的网络浏览器要求太高,还是我做错了什么?有办法解决吗?我的整个模特是错的吗?
可能的解决方案
我摆弄一些想法,但没有一个真正起作用。我尝试强制后端一次不输出超过200个事件,并在所有当前队列都已处理完毕后添加新的轮询请求。仍然有堆栈溢出。我还尝试在完成处理后删除eventQueue
对象(即使当时为空)并重新创建它,以希望它可以释放一些底层内存或其他东西。我缺少创意,因此任何创意,指针或一般建议都将不胜感激。
编辑:
我有一个启示!我想我完全知道为什么所有这些事情都会发生(但是我仍然不确定如何处理和修复它),我还将提供一些基本的代码摘录。
var eventQueue = new Object();
var processingEvent = false;
var lastRetrievedEventId = 0;
var currentEventId = 0;
function sendPoll() {
// Standard jsonp request (to a intentionally slow backend, i.e. long-polling),
// callback set to pollCallback(). Provide currentEventId to the server to only get
// the events starting from that point.
}
function pollCallback( data ) {
// Make sure the data isn't empty, this happens if the jsonp request
// expires (30s in my case) and it didn't get any new data.
if( !jQuery.isEmptyObject( data ) )
{
// Add each new event to the event queue.
$.each(data.events, function() {
eventQueue[ this.id ] = this;
lastRetrievedEventId = this.id; // Since we just put the event in the queue, we know it is officially the last one "retrieved".
});
// Process the next event, we know there has to be at least one in queue!
processNextEvent();
}
// Go look for new events!
sendPoll();
}
function processNextEvent() {
// Do not process events if they are currently being processed, that would happen
// when an event contains an asynchronous function, like an AJAX call.
if( !processingEvent )
{
var nextEventId = currentEventId + 1;
// Before accessing it directly, make sure the "next event" is in the queue.
if( Object.prototype.hasOwnProperty.call(eventQueue, nextEventId) )
{
processingEvent = true;
processEvent( eventQueue[ nextEventId ] );
}
}
}
function processEvent( event ) {
// Do different actions based on the event type.
switch( event.eventType ) {
case SOME_TYPE:
// Do stuff pertaining to SOME_TYPE.
eventComplete( event );
break;
case SOME_OTHER_TYPE:
// Do stuff pertaining to SOME_OTHER_TYPE.
eventComplete( event );
break;
// Etc. Many more cases here. If there is an AJAX call,
// the eventComplete( event ) is placed in the success: callback
// of that AJAX call, I do not want events to be processed in the wrong order.
}
}
function eventComplete( event ) {
// The event has completed, time to process the event after it.
currentEventId = event.id; // Since it was fully processed, it is now the most current event.
delete eventQueue[ event.id ]; // It was fully processed, we don't need it anymore.
processingEvent = false;
processNextEvent(); // Process the next event in queue. Most likely the source of all my woes.
}
function myApplicationIsReady() {
// The DOM is fully loaded, my application has initiated all its data and variables,
// start the long polling.
sendPoll();
}
$(function() {
// Initializing my application.
myApplicationIsReady();
});
myApplicationIsReady() -> sendPoll()
pollCallback() -> [ processNextEvent() -> processEvent() -> eventComplete() -> processNextEvent() ]
pollCallback() -> processNextEvent() -> processEvent() -> eventComplete() -> sendPoll()
最佳答案
不使用递归调用函数,而是使用setTimeout(func, 0)
怎么样?
关于javascript - 事件系统导致Javascript中的堆栈溢出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11399322/
我有这个代码: System.err.print("number of terms = "); System.out.println(allTerms.size()); System.err
我有以下问题:在操作系统是 Linux 的情况下和在操作系统是 MacOs 的情况下,我必须执行不同的操作。 所以我创建了以下 Ant 脚本目标: /u
我正在调用 system("bash ../tools/bashScript\"This is an argument!\"&"),然后我正在调用 close(socketFD) 直接在 system
使用最初生成的随机元素来约束随机数组的连续元素是否有效。 例如:我想生成一组 10 个 addr、size 对来模拟典型的内存分配例程并具有如下类: class abc; rand bit[5:0
我正在创建一个必须使用system(const char*)函数来完成一些“繁重工作”的应用程序,并且我需要能够为用户提供粗略的进度百分比。例如,如果操作系统正在为您移动文件,它会为您提供一个进度条,
我即将编写一些项目经理、开发人员和业务分析师会使用的标准/指南和模板。目标是更好地理解正在开发或已经开发的解决方案。 其中一部分是提供有关记录解决方案的标准/指南。例如。记录解决/满足业务案例/用户需
在开发使用压缩磁盘索引或磁盘文件的应用程序时,其中部分索引或文件被重复访问(为了论证,让我们说一些类似于 Zipfian 分布的东西),我想知道什么时候足够/更好地依赖操作系统级缓存(例如,Debia
我们编写了一个 powershell 脚本,用于处理来自内部系统的图像并将其发送到另一个系统。现在,业务的另一部分希望加入其中,对数据进行自己的处理,并将其推送到另一个系统。打听了一下,公司周围有几个
我正在尝试朗姆酒我的应用程序,但我收到以下错误:System.Web.HttpUnhandledException:引发了“System.Web.HttpUnhandledException”类型的异
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,
所以我在其他程序中没有收到此错误,但我在这个程序中收到了它。 这个程序是一个我没有收到错误的示例。 #include int main() { system("pause"); } // en
我在 c# System.URI.FormatExption 中遇到问题 为了清楚起见,我使用的是 Segseuil 的 Matlab 方法,并且它返回一个图片路径 result。我想为其他用户保存此
我正在尝试像这样设置文本框的背景色: txtCompanyName.BackColor = Drawing.Color.WhiteSmoke; 它不喜欢它,因为它要我在前面添加系统,例如: txtCo
请帮助我解决 System.StackOverflowException我想用 .aspx 将记录写入数据库我使用 4 层架构来实现这一切都正常但是当我编译页面然后它显示要插入数据的字段时,当我将数据
我使用了一些通常由系统调用的API。 因此,我将 android:sharedUserId="android.uid.system" 添加到 manifest.xml, 并使用来自 GIT 的 And
我正在尝试创建一个小型应用程序,它需要对/system 文件夹进行读/写访问(它正在尝试删除一个文件,并创建一个新文件来代替它)。我可以使用 adb 毫无问题地重新挂载该文件夹,如果我这样做,我的应用
我想从没有 su 的系统 priv-app 将/system 重新挂载为 RW。如何以编程方式执行此操作?只会用 Runtime.getruntime().exec() 执行一个 shell 命令吗
我正在尝试制作一个带有登录系统的程序我对此很陌生,但我已经连续工作 8 个小时试图解决这个问题。这是我得到的错误代码 + ServerVersion 'con.ServerVersion' threw
当我“构建并运行”Code::Blocks 中的程序时,它运行得非常好!但是当我从“/bin”文件夹手动运行它时,当它试图用 system() 调用“temp.bat”时,它会重置。这是为什么?它没有
我想使用 system/pipe 命令来执行具有特殊字符的命令。下面是示例代码。通过系统/管道执行命令后,它通过改变特殊字符来改变命令。我很惊讶地看到系统命令正在更改作为命令传递的文本。 run(ch
我是一名优秀的程序员,十分优秀!