专栏首页前端杂货铺node中子进程同步输出

node中子进程同步输出

管道

通过“child_process”模块fork出来的子进程都是返回一个ChildProcess对象实例,ChildProcess类比较特殊无法手动创建该对象实例,只能使用fork或者spawn,而且与process对象不同的是,ChildProcess实例的stdin为可写流,stdout和stderr为可读流。因此通过childprocess.stdin可以输入数据,通过childprocess.stdout可将子进程的数据数据输出到父进程中。

具体实现
var child = require('child_process')
    , fs = require('fs');
var childProcess = child.exec(cmd, {env: process.env, maxBuffer: 20*1024*1024}, function(err) {
  });
var stdoutStream = fs.createWriteStream(escape(stdoutFilePath));
  childProcess.stdout.pipe(stdoutStream, {end: false});
  childProcess.stderr.pipe(stdoutStream, {end: false});
  // 使用node的后压机制
  childProcess.stdout.pipe(process.stdout);
  childProcess.stderr.pipe(process.stderr);
  var stdoutEnded = false, stderrEnded = false;

  function tryClosing(){ if(stdoutEnded && stderrEnded){ stdoutStream.end(); } }
  childProcess.stdout.on('end', function(){ stdoutEnded = true; tryClosing(); });
  childProcess.stderr.on('end', function(){ stderrEnded = true; tryClosing(); });

这种方式适用于大多数场景,直接使用流特性完成子进程数据的输出。

文件检测

在某些系统的node环境下,“child_process”并未提供execSync特性,因此需要hack,这里参考shelljs的实现机制。

使用系统兼容较好的exec函数完成基本功能,在shell命令执行完毕后写入状态信息到某些临时文件,最后通过循环不断读取新写入该临时文件的数据。由于在shell命令执行过程中需要模拟同步效果,因此在循环中不仅仅获取新写入的数据,同时需要模拟I/O阻塞操作,此处shelljs的作者通过尝试所有的同步IO API,发现fs.writeFileSync操作可以较少的减轻CPU利用率,因此使用该函数阻塞I/O。

具体实现
var child = require('child_process')
    , fs = require('fs');
var childProcess = child.exec('"+escape(cmd)+"', {env: process.env, maxBuffer: 20*1024*1024}, function(err) {
  fs.writeFileSync('"+escape(codeFile)+"', err ? err.code.toString() : '0');
  });

var previousStdoutContent = '';
  // Echoes stdout changes from running process, if not silent
  function updateStdout() {
    if (options.silent || !fs.existsSync(stdoutFile))
      return;

    var stdoutContent = fs.readFileSync(stdoutFile, 'utf8');
    // No changes since last time?
    if (stdoutContent.length <= previousStdoutContent.length)
      return;

    process.stdout.write(stdoutContent.substr(previousStdoutContent.length));
    previousStdoutContent = stdoutContent;
  }

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 自动补全搜索实现

    目前大多数搜索框都已实现自动补全功能,自己也私底下实现了一个简易版本, 在此总结过程中的一些要点:   1,侦听文本框的value值改变,注意在Ie8及其之前版...

    欲休
  • 使用yield进行异步流程控制

    现状 目前我们对异步回调的解决方案有这么几种:回调,deferred/promise和事件触发。回调的方式自不必说,需要硬编码调用,而且有可能会出现复杂的嵌套关...

    欲休
  • 样式化加载失败的图片

    本片文章翻译自 Styling Broken Images 翻译过程中可能会在原意不变的基础上有些细微改动,望读者见谅 加载失败的图片是比较丑陋的,比如 但是我...

    欲休
  • day5-json和pickle序列化

    一.json模块 序列化:把一个对象的形态改变一下,使他能够存放在文件中,或者在网络上传输,序列化也叫持久化,是把对象存储到永久介质中,这样就不会因为掉电而丢失...

    用户1679793
  • 例说七层OSI参考模型

    OSI并不是一个协议,也不是一项技术,是一个参考模型。OSI(Open SystemInterconnect),即开放式系统互联。 一般都叫OSI参考模型,是I...

    碎碎思
  • 巧用TagAlign格式来进行ATAC中的shift reads操作

    由于Tn5转座酶的特性,在ATAC数据分析中,首选需要对bam文件中reads的比对位置进行shift, 然后再进行peak calling。那么如何进行这一操...

    生信修炼手册
  • Java并发编程实战 01并发编程的Bug源头

    编写正确的并发程序对我来说是一件极其困难的事情,由于知识不足,只知道synchronized这个修饰符进行同步。 本文为学习极客时间:Java并发编程实战 0...

    Johnson木木
  • Java多线程—AQS框架源码阅读

    AQS,全称AbstractQueuedSynchronizer,是Concurrent包锁的核心,没有AQS就没有Java的Concurrent包。

    Zack说码
  • python模块之json

    JSON(JavaScript Object Notation, JS 对象标记) 是一种轻量级的数据交换格式。JSON的数据格式其实就是python里面的字典...

    菲宇
  • 快速学习-Linux文档的查看指令

    作用:查看一个文件的末n行 语法:#tail -n 文件的路径 说明:-n可以不写,不写,默认表示10行。 案例:使用tail指令查看root/insta...

    cwl_java

扫码关注云+社区

领取腾讯云代金券