如何防止ChildProcess耗尽内存?电子/ Node.js的

时间:2018-02-25 18:25:44

标签: javascript node.js linux electron

使用Electron和Node.js为生成数据的进程编写简单的用户界面,然后允许用户调用gnuplot来显示数据。这是main.js中调用gnuplot(gnuplot5-qt)的代码。

var menu = Menu.buildFromTemplate([
  {
      label: 'Run Graph',
      click() {
        commandLine()
      }
  },
// Other code
function commandLine () {
    var child = require('child_process').execFile;
    var executablePath = "/usr/bin/gnuplot";
    var parameters = ["-p","-e","filename='/home/prog1/PP_Logs/log1.txt'","/home/prog1/plot_log.p"];

    child(executablePath, parameters, function(err, data) {
      console.log(err);
      console.log(data.toString());
    });

gnuplot总是打开,有时会保持打开状态,我可以正常退出,但随机打开后会立即关闭并打印出以下错误:

{ Error: Command failed: /usr/bin/gnuplot -p -e filename='/home/prog1/PP_Logs/log1.txt' /home/prog1/plot_log.p

at ChildProcess.exithandler (child_process.js:282:12)
at emitTwo (events.js:125:13)
at ChildProcess.emit (events.js:213:7)
at maybeClose (internal/child_process.js:921:16)
at Socket.stream.socket.on (internal/child_process.js:348:11)
at emitOne (events.js:115:13)
at Socket.emit (events.js:210:7)
at Pipe._handle.close [as _onclose] (net.js:549:12)
killed: false,
code: null,
signal: 'SIGSEGV',
cmd: '/usr/bin/gnuplot -p -e filename=\'/home/prog1/PP_Logs/log1.txt\' /home/prog1/plot_log.p' }

如果启动gimp而不是gnuplot,那么相同的代码可以正常运行:

 var executablePath = "/usr/bin/gimp";
 var parameters = ["-s","/home/geenweb/Pictures/lgcm110.jpg"];

鉴于SIGSEGV,我假设gnuplot在崩溃时内存不足。有没有办法分配更多的内存?有没有更好的方法来调用gnuplot?我只是在学习Electron,Node.js和javascript。谢谢你的帮助。

2 个答案:

答案 0 :(得分:2)

首先,您应该确保运行当前的LTS版本的Node,在撰写本文时是v8.9.4。

如果您正在使用Electron提供自己的NodeJS运行时,您应该仍然保持Electron和本地版本的NodeJS尽可能最新。

默认情况下,

ChildProcess.execFile()会在新进程中运行您的命令,这可能是导致错误的原因。您可能希望通过将传递给shell的选项中的execFile()标记设置为true来指定您的命令在新生成的shell中运行,这将导致默认{{1}要使用的shell或要使用的shell的路径。

/bin/sh/bin/sh shell中运行您的进程将允许您访问shell /bin/bash内置命令,该命令将为您提供修改shell资源限制的方法。有关ulimit命令以及如何使用它的信息,请参阅Setting Limits With ulimit

如果是这种情况,您需要将命令调用包装在首先运行ulimit且具有正确限制的shell脚本中。

有关此主题的进一步讨论,请参阅Limit Memory Usage For A Single Linux Process

您的问题也可能是由用于通过ulimitstdout传输运行命令输出的缓冲区的默认分配大小引起的。

如果是这种情况并且程序的输出超过stderr字节,则可以使用传递给200*1024的选项中的maxBuffer来增加这些缓冲区的大小。

答案 1 :(得分:1)

感谢。我使用的是Node v8.9.4。

我认为gnuplot从shell启动时工作正常,但我在阅读你的回复后尝试重复启动。果然,我每次运行都会遇到分段违规。我用Google搜索并发现:https://sourceforge.net/p/gnuplot/bugs/1885/我将脚本文件中的终端从wxt更改为qt,现在我不再从命令行发生崩溃,或通过电子启动它。谢谢你给我提示我需要在正确的地方看看。