当前位置: 首页 > 面试题库 >

为什么while循环会阻止事件循环?

宋晋
2023-03-14
问题内容

以下示例在Node.js书中给出:

var open = false;

setTimeout(function() {
  open = true
}, 1000)

while (!open) {
  console.log('wait');
}

console.log('open sesame');

解释了while循环为何阻止执行时,作者说:

节点将永远不会执行超时回调,因为事件循环卡在了循环中,而循环在第7行开始了,因此永远不会给它处理超时事件的机会!

但是,作者没有解释为什么这是在事件循环的背景下发生的,还是在幕后真正发生了什么。

有人可以详细说明吗?为什么节点卡住?以及如何在保留while控制结构的同时更改上述代码,以使事件循环不会被阻塞,并且代码将按人们合理预期的方式运行;在setTimeout发生火灾之前,将仅记录1秒钟的等待时间,然后在记录“打开芝麻”后退出该过程。


问题答案:

真的很简单。在内部,node.js包含以下类型的循环:

  • 从事件队列中获取一些东西
  • 运行指示的任何任务并运行直到返回
  • 完成上述任务后,从事件队列中获取下一个项目
  • 运行指示的任何任务并运行直到返回
  • 反复冲洗,起泡,重复

如果在某个时刻事件队列中什么都没有,则进入睡眠状态直到事件队列中放置了某些东西。

因此,如果一段Javascript处于while()循环中,则该任务未完成,按照上述顺序,在事件队列中没有新东西会被拾取,直到之前的任务完全完成为止。因此,一个很长或永远运行的while()循环只会使工作变得困难。因为Javascript一次只运行一个任务(用于JS执行的单线程),所以如果一个任务在while循环中旋转,那么其他任何事情都将无法执行。

这是一个简单的示例,可能有助于解释它:

 var done = false;

 // set a timer for 1 second from now to set done to true
 setTimeout(function() {
      done = true;
 }, 1000);

 // spin wait for the done value to change
 while (!done) { /* do nothing */}

 console.log("finally, the done value changed!");

从逻辑上讲,有些人可能会认为while循环将旋转直到计时器启动,然后计时器将更改doneto
的值,true然后while循环将完成,最后的while
console.log()将执行。那不会发生。实际上,这将是一个无限循环,并且该console.log()语句将永远不会执行。

问题是,一旦您进入while()循环等待循环,就无法再执行其他Javascript。因此,想要更改变量值的计时器done无法执行。因此,while循环条件永远不会改变,因此它是一个无限循环。

这是JS引擎内部发生的事情:

  1. done 变量初始化为 false
  2. setTimeout() 从现在开始将计时器事件安排为1秒
  3. while循环开始旋转
  4. 在while循环旋转1秒后,计时器会在内部触发JS引擎,并将计时器回调添加到事件队列中。这可能发生在JS引擎内部的其他线程上。
  5. while循环保持旋转,因为done变量永不改变。因为它继续旋转,所以JS引擎永远不会完成该执行线程,也永远不会从事件队列中提取下一个项目。

node.js是一个事件驱动的环境。为了在现实世界的应用程序中解决此问题,该done标志将在将来的某个事件中更改。因此,while您将来可以为某个相关事件注册事件处理程序,而不是自旋循环,然后在那里进行工作。在绝对最坏的情况下,您可以设置一个循环计时器和“html" target="_blank">轮询”来经常检查该标志,但是在几乎每种情况下,您都可以为实际事件注册一个事件处理程序,该事件处理程序将导致该done标志发生更改并执行您的工作。经过适当设计的代码可以知道其他代码想要知道何时发生了更改,甚至可以提供自己的事件侦听器和自己的通知事件,以便人们可以注册某个兴趣,甚至只是简单的回调。



 类似资料:
  • 本文向大家介绍什么是事件循环?相关面试题,主要包含被问及什么是事件循环?时的应答技巧和注意事项,需要的朋友参考一下 Node采用的是单线程的处理机制(所有的I/O请求都采用非阻塞的工作方式),至少从Node.js开发者的角度是这样的。而在底层,Node.js借助libuv来作为抽象封装层,从而屏蔽不同操作系统的差异,Node可以借助livuv来实现线程。下图表示Node和libuv的关系 Libu

  • 我有一个看起来很简单的问题,但由于某种原因我无法绕过它。基本上我的程序正在导致一个无限循环,我不知道为什么。 下面是我陷入的特定循环: 当我运行它时,它总是问我输入列#。我给它一个数字,它接受这个数字,$response变为True,但while循环继续运行,就好像<code>的$response</code>为false一样。我是Perl新手,所以可能我遗漏了一些东西,但是($response=

  • 我想我的程序跳过了while循环,但我真的不确定到底发生了什么。该函数应该通过找到GCD,然后将分子和分母除以该数字来减少分数。 我得到分子和分母的绝对值,以确保如果分数是负数,我会在最后保持它。如果分子为0,则要求我返回(0,1)。问题是关于while循环。。。似乎它被完全跳过了。有什么建议吗?

  • 我想写一个程序,列出所有的。xml文件,然后将其复制到另一个目录并从原始目录中删除。程序的这一部分运行良好。我想这样做,我可以点击GUI中的一个按钮,让它扫描和处理文件夹,直到我按下一个按钮将其关闭。再说一次,打开它不是问题,但试图阻止它让我难堪。我想在两者之间等待一段时间,但要利用时间。sleep(x)冻结整个程序,在它停止Hibernate之前不允许我输入任何命令,只允许它处理然后再次Hibe

  • 我正在运行一个纸牌游戏,两个玩家轮流挑选纸牌,玩家向前移动许多空格。在这种情况下,一旦任何一个玩家达到25或更大的数字,他们击中了家,游戏应该停止。 我的while循环在两个玩家都达到25后继续循环,即使我已经设置了条件,在玩家1的值小于25或玩家2的值小于25时循环。 当循环条件出了什么问题? 编辑:这是我运行的一个测试的一些输出。

  • 只要给定条件为真,Perl编程语言中的while循环语句就会重复执行目标语句。 语法 (Syntax) Perl编程语言中while循环的语法是 - while(condition) { statement(s); } 这里的statement(s)可以是单个陈述或一个陈述块。 condition可以是任何表达。 当条件为真时,循环迭代。 当条件变为假时,程序控制将立即传递到循环之后的行。