草庐IT

javascript - JS : Maximum call stack size exceeded on disconnect despite function being empty

coder 2023-09-18 原文

我对 js 有点陌生,正在尝试使用 net 模块。

我有一个简单的服务器在运行,它将包分发给所有已知的客户端,但是一旦一个客户端断开连接,服务器就会在“结束”事件上崩溃,尽管这个功能只包括一个日志命令:

var clients = [];

const server = net.createServer((c) => {

  // 'connection' listener
  console.log("client connected")
  clients.push(c);

  c.on('end', () => {
    console.log('client disconnected');
    //dc(c)
  });

  c.on('error', () => {
    c.write("400");
  })

  c.on('data', (data) => {

    console.log(data.toString())
    writeAll(c, data);

  });

});

Write all 是下面的一个函数,它将传入的包分发给所有已知的客户端,不包括 c:

function writeAll(exclude, buffer) {

  clients.forEach((client) => {
    if(client != exclude) {
      client.write(buffer);
    }
  });

}

dc() 是一个函数,它应该从我之前定义的数组列表中删除客户端,但被注释掉了,所以它应该无关紧要。 一切正常,直到一个客户端断开连接,即当我收到以下错误消息时:

internal/errors.js:207
function getMessage(key, args) {
                   ^

RangeError: Maximum call stack size exceeded
    at getMessage (internal/errors.js:207:20)
    at new NodeError (internal/errors.js:153:13)
    at doWrite (_stream_writable.js:411:19)
    at writeOrBuffer (_stream_writable.js:399:5)
    at Socket.Writable.write (_stream_writable.js:299:11)
    at Socket.c.on (C:\Users\...\server.js:17:7)
    at Socket.emit (events.js:197:13)
    at errorOrDestroy (internal/streams/destroy.js:98:12)
    at onwriteError (_stream_writable.js:430:5)
    at onwrite (_stream_writable.js:461:5)

引用的第 17 行是您在上面看到的 on-error 事件。 我已经尝试将最大堆栈长度扩展到 2000,但它仍然不起作用。最奇怪的是,抛出错误消息的部分是从另一个我知道有效的代码复制粘贴过来的。

如果你们中的一位专业人士愿意接受这个,我将不胜感激,因为我完全迷失了。

编辑:完整文件可以在这里找到:https://ghostbin.com/paste/knm3f

编辑 2:我重新安装了 net 模块,现在错误发生了变化,让我更加困惑:

internal/errors.js:222
  const expectedLength = (msg.match(/%[dfijoOs]/g) || []).length;
                              ^
RangeError: Maximum call stack size exceeded
    at String.match (<anonymous>)
    at getMessage (internal/errors.js:222:31)
    at new NodeError (internal/errors.js:153:13)
    at doWrite (_stream_writable.js:411:19)
    at writeOrBuffer (_stream_writable.js:399:5)
    at Socket.Writable.write (_stream_writable.js:299:11)
    at Socket.c.on (C:\Users\...\server.js:17:7)
    at Socket.emit (events.js:197:13)
    at errorOrDestroy (internal/streams/destroy.js:98:12)
    at onwriteError (_stream_writable.js:430:5)

编辑 3:确实有效的代码已经过测试,由于某种原因,即使在编写它的机器上以及我的笔记本电脑上也会遇到同样的问题。我感觉这不适合我。

编辑 4:SOOOOOOOOOOO... 事实证明,Microsoft 再次将拳头放在了我的后端。它在 Linux 发行版上运行良好 100。我会把赌注押在 Windows 防火墙更改或我不同意的其他更新上。谢谢微软。你是在逼我迁移到 Linux。

编辑 5:断开连接的错误仅在 Windows 上抛出,但 stackoverflow 是由写入客户端的错误事件引起的,它本身抛出错误,因为客户端不再连接。

最佳答案

您的代码中有两个错误,它们绝对不是太明显。

无论何时向客户端写入,都使用 writeAll() 中的循环:

clients.forEach((client) => {
  if(client != exclude) {
    client.write(buffer);
  }
});

在这里我们看到您执行了 client.write(...)。该命令可能会检测到此特定 client 已关闭其连接。这意味着 emit('error') 将会发生,您的相应回调将被调用。

在那个回调中,你正在调用 dc(c),它应该从数组中删除客户端:

clients.splice(index, 1)

换句话说,您在执行 clients.forEach() 时正在修改名为 clients 的数组。这是一个很大的禁忌!实际上,我试图完全忘记 forEach() 函数。它给我带来了很多问题!有人说 forEach() 也比 for()while() 循环慢。但这个问题可能已经得到改善。

至少,您要做的是使用 for() 循环:

for(let i = 0; i < clients.length; ++i)
{
   ...
}

请注意,我没有先将 clients.length 保存在变量中,因为它可能会随时间变化!

这种方法的问题在于,无论何时发生 splice(),您都会错过一些客户。

下一个最好的办法是使用向后循环。为此,while() 非常适用:

let i = clients.length
while(i > 0)
{
    --i
    ...
}

只要满足以下条件,这个循环就会工作得更好:(1) 只有一个项目被 splice() 输出,(2) 没有新的客户端被添加到列表中。

我建议的最后一个强有力的解决方案是首先复制数组:

const client_list = clients.slice()
client_list.forEach((client) => { ... })    // if you really want to use the forEach()?!

副本不会改变:它是本地的并且const

但为什么堆栈会因此变得困惑?可能是因为错误导致您对客户端执行了 write() ......好吧,这与 forEach() 无关,但此代码循环:

c.on('error', () => {
    c.write("400")     // are you missing a "\n", btw?
});

如您所见,c.write() 是产生错误的罪魁祸首,您将在这个 on() 中永远循环回调...您需要检查 c 是否有效或捕获错误:

c.on('error', () => {
    try
    {
        c.write("400")     // are you missing a "\n", btw?
    }
    catch(err)
    {
        // maybe log the error?
        console.log(err)
    }
});

也就是说,我从经验中知道,在我从我的代码中删除所有 forEach() 调用并制作数组(对象)的副本后,我无法确定它们是否不会得到在我的脚下修改后,我的代码运行得更好。

关于javascript - JS : Maximum call stack size exceeded on disconnect despite function being empty,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54694779/

有关javascript - JS : Maximum call stack size exceeded on disconnect despite function being empty的更多相关文章

  1. ruby-on-rails - 使用 javascript 更改数据方法不会更改 ajax 调用用户的什么方法? - 2

    我遇到了一个非常奇怪的问题,我很难解决。在我看来,我有一个与data-remote="true"和data-method="delete"的链接。当我单击该链接时,我可以看到对我的Rails服务器的DELETE请求。返回的JS代码会更改此链接的属性,其中包括href和data-method。再次单击此链接后,我的服务器收到了对新href的请求,但使用的是旧的data-method,即使我已将其从DELETE到POST(它仍然发送一个DELETE请求)。但是,如果我刷新页面,HTML与"new"HTML相同(随返回的JS发生变化),但它实际上发送了正确的请求类型。这就是这个问题令我困惑的

  2. ruby-on-rails - Assets 管道损坏 : Not compiling on the fly css and js files - 2

    我开始了一个新的Rails3.2.5项目,Assets管道不再工作了。CSS和Javascript文件不再编译。这是尝试生成Assets时日志的输出:StartedGET"/assets/application.css?body=1"for127.0.0.1at2012-06-1623:59:11-0700Servedasset/application.css-200OK(0ms)[2012-06-1623:59:11]ERRORNoMethodError:undefinedmethod`each'fornil:NilClass/Users/greg/.rbenv/versions/1

  3. ruby-on-rails - Rails - 理解 application.js 和 application.css - 2

    rails新手。只是想了解\assests目录中的这两个文件。例如,application.js文件有如下行://=requirejquery//=requirejquery_ujs//=require_tree.我理解require_tree。只是将所有JS文件添加到当前目录中。根据上下文,我可以看出requirejquery添加了jQuery库。但是它从哪里得到这些jQuery库呢?我没有在我的Assets文件夹中看到任何jquery.js文件——或者直接在我的整个应用程序中没有看到任何jquery.js文件?同样,我正在按照一些说明安装TwitterBootstrap(http:

  4. ruby - 在 Mechanize 中使用 JavaScript 单击链接 - 2

    我有这个:AccountSummary我想单击该链接,但在使用link_to时出现错误。我试过:bot.click(page.link_with(:href=>/menu_home/))bot.click(page.link_with(:class=>'top_level_active'))bot.click(page.link_with(:href=>/AccountSummary/))我得到的错误是:NoMethodError:nil:NilClass的未定义方法“[]” 最佳答案 那是一个javascript链接。Mechan

  5. node.js - 如何在 Travis CI 上的一个项目中运行 Node.js 和 Ruby 测试 - 2

    我有一个包含多个组件的存储库,其中大部分是用JavaScript(Node.js)编写的,一个是用Ruby(RubyonRails)编写的。我想要一个.travis.yml文件来触发一个运行每个组件的所有测试的构建。根据thisTravisCIGoogleGroupthread,目前还没有官方支持。我的目录结构是这样的:.├──构建服务器├──核心├──扩展├──网络应用├──流浪文件├──package.json├──.travis.yml└──生成文件我希望能够运行特定版本的Ruby(2.2.2)和Node.js(0.12.2)。我已经有了一个make目标,所以maketest在每

  6. javascript - jQuery 的 jquery-1.10.2.min.map 正在触发 404(未找到) - 2

    我看到有关未找到文件min.map的错误消息:GETjQuery'sjquery-1.10.2.min.mapistriggeringa404(NotFound)截图这是从哪里来的? 最佳答案 如果ChromeDevTools报告.map文件的404(可能是jquery-1.10.2.min.map、jquery.min.map或jquery-2.0.3.min.map,但任何事情都可能发生)首先要知道的是,这仅在使用DevTools时才会请求。您的用户不会遇到此404。现在您可以修复此问题或禁用sourcemap功能。修复:获取文

  7. ruby-on-rails - 我将 Rails3 与 tinymce 一起使用。如何呈现用户关闭浏览器javascript然后输入xss? - 2

    我有一个用Rails3编写的站点。我的帖子模型有一个名为“内容”的文本列。在帖子面板中,html表单使用tinymce将“content”列设置为textarea字段。在首页,因为使用了tinymce,post.html.erb的代码需要用这样的原始方法来实现。.好的,现在如果我关闭浏览器javascript,这个文本区域可以在没有tinymce的情况下输入,也许用户会输入任何xss,比如alert('xss');.我的前台会显示那个警告框。我尝试sanitize(@post.content)在posts_controller中,但sanitize方法将相互过滤tinymce样式。例如

  8. node.js - 从未编写过任何自动化测试,我应该如何开始行为驱动开发? - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。多年来,我一直在使用多种语言进行编程,并且认为自己总体上相当擅长。但是,我从未编写过任何自动化测试:没有单元测试,没有TDD,没有BDD,什么都没有。我已经尝试开始为我的项目编写适当的测试套件。我可以看到在进行任何更改后能够自动测试项目中所有代码的理论值(value)。我可以看到像RSpec和Mocha这样的测试框架应该如何使设置和运行所述测试变得相当容易

  9. ruby - 使用 Selenium WebDriver 启用/禁用 javascript - 2

    出于某种原因,我必须为Firefox禁用javascript(手动,我们按照提到的步骤执行http://support.mozilla.org/en-US/kb/javascript-settings-for-interactive-web-pages#w_enabling-and-disabling-javascript)。使用Ruby的SeleniumWebDriver如何实现这一点? 最佳答案 是的,这是可能的。而是另一种方式。您首先需要查看链接Selenium::WebDriver::Firefox::Profile#[]=

  10. ruby-on-rails - 将 Angular JS 与 Rails 集成 - 2

    我需要一些指导来了解如何将Angular整合到rails中。选择Rails的原因:我喜欢他们偏执的做事方式。还有迁移,gem真的很酷。使用angular的原因:我正在研究和寻找最适合SPA的框架。Backbone似乎太抽象了。我不得不在Angular和Ember之间做出选择。我首先开始阅读Angular,它对我来说很有意义。所以我从来没有去读过关于ember的文章。使用Angular和Rails的原因:我研究并尝试使用小型框架,例如grape、slim(是的,我也使用php)。但我觉得需要坚持项目的长期范围。我个人喜欢用Rails的方式做事。这就是我需要帮助的地方,我在Rails4中有

随机推荐