详解Node.JS模块 process


Posted in Javascript onAugust 31, 2020

process 模块是 nodejs 提供给开发者用来和当前进程交互的工具,它的提供了很多实用的 API。从文档出发,管中窥豹,进一步认识和学习 process 模块:

  • 如何处理命令参数?
  • 如何处理工作目录?
  • 如何处理异常?
  • 如何处理进程退出?
  • process 的标准流对象
  • 深入理解 process.nextTick

如何处理命令参数?

命令行参数指的是 2 个方面:

  • 传给 node 的参数。例如 node --harmony script.js --version 中,--harmony 就是传给 node 的参数
  • 传给进程的参数。例如 node script.js --version --help 中,--version --help 就是传给进程的参数

它们分别通过 process.argvprocess.execArgv 来获得。

如何处理工作目录?

通过process.cwd()可以获取当前的工作目录。

通过process.chdir(directory)可以切换当前的工作目录,失败后会抛出异常。实践如下:

function safeChdir(dir) {
 try {
  process.chdir(dir);
  return true;
 } catch (error) {
  return false;
 }
}

如何处理异常?

uncaughtException 事件

Nodejs 可以通过 try-catch 来捕获异常。如果异常未捕获,则会一直从底向事件循环冒泡。如是冒泡到事件循环的异常没被处理,那么就会导致当前进程异常退出。

根据文档,可以通过监听 process 的 uncaughtException 事件,来处理未捕获的异常:

process.on("uncaughtException", (err, origin) => {
 console.log(err.message);
});

const a = 1 / b;
console.log("abc"); // 不会执行

上面的代码,控制台的输出是:b is not defined。捕获了错误信息,并且进程以0退出。开发者可以在 uncaughtException 事件中,清除一些已经分配的资源(文件描述符、句柄等),不推荐在其中重启进程。

unhandledRejection 事件

如果一个 Promise 回调的异常没有被.catch()捕获,那么就会触发 process 的 unhandledRejection 事件:

process.on("unhandledRejection", (err, promise) => {
 console.log(err.message);
});

Promise.reject(new Error("错误信息")); // 未被catch捕获的异常,交由unhandledRejection事件处理

warning 事件

告警不是 Node.js 和 Javascript 错误处理流程的正式组成部分。 一旦探测到可能导致应用性能问题,缺陷或安全隐患相关的代码实践,Node.js 就可发出告警。

比如前一段代码中,如果出现未被捕获的 promise 回调的异常,那么就会触发 warning 事件。

如何处理进程退出?

process.exit() vs process.exitCode

一个 nodejs 进程,可以通过 process.exit() 来指定退出代码,直接退出。不推荐直接使用 process.exit(),这会导致事件循环中的任务直接不被处理,以及可能导致数据的截断和丢失(例如 stdout 的写入)。

setTimeout(() => {
 console.log("我不会执行");
});

process.exit(0);

正确安全的处理是,设置 process.exitCode,并允许进程自然退出。

setTimeout(() => {
 console.log("我不会执行");
});

process.exitCode = 1;

beforeExit 事件

用于处理进程退出的事件有:beforeExit 事件 和 exit 事件。

当 Node.js 清空其事件循环并且没有其他工作要安排时,会触发 beforeExit 事件。例如在退出前需要一些异步操作,那么可以写在 beforeExit 事件中:

let hasSend = false;
process.on("beforeExit", () => {
 if (hasSend) return; // 避免死循环

 setTimeout(() => {
  console.log("mock send data to serve");
  hasSend = true;
 }, 500);
});

console.log(".......");
// 输出:
// .......
// mock send data to serve

注意:在 beforeExit 事件中如果是异步任务,那么又会被添加到任务队列。此时,任务队列完成所有任务后,又回触发 beforeExit 事件。因此,不处理的话,可能出现死循环的情况。如果是显式调用 exit(),那么不会触发此事件。

exit 事件

在 exit 事件中,只能执行同步操作。在调用 ‘exit' 事件监听器之后,Node.js 进程将立即退出,从而导致在事件循环中仍排队的任何其他工作被放弃。

process 的标准流对象

process 提供了 3 个标准流。需要注意的是,它们有些在某些时候是同步阻塞的(请见文档)。

  • process.stderr:WriteStream 类型,console.error的底层实现,默认对应屏幕
  • process.stdout:WriteStream 类型,console.log的底层实现,默认对应屏幕
  • process.stdin:ReadStream 类型,默认对应键盘输入

下面是基于“生产者-消费者模型”的读取控制台输入并且及时输出的代码:

process.stdin.setEncoding("utf8");

process.stdin.on("readable", () => {
 let chunk;
 while ((chunk = process.stdin.read()) !== null) {
  process.stdout.write(`>>> ${chunk}`);
 }
});

process.stdin.on("end", () => {
 process.stdout.write("结束");
});

关于事件的含义,还是请看stream 的文档。

深入理解 process.nextTick

我第一次看到 process.nextTick 的时候是比较懵的,看文档可以知道,它的用途是:把回调函数作为微任务,放入事件循环的任务队列中。但这么做的意义是什么呢?

因为 nodejs 并不适合计算密集型的应用,一个进程就一个线程,在当下时间点上,就一个事件在执行。那么,如果我们的事件占用了很多 cpu 时间,那么之后的事件就要等待非常久。所以,nodejs 的一个编程原则是尽量缩短每一个事件的执行事件。process.nextTick 的作用就在这,将一个大的任务分解成多个小的任务。示例代码如下:

// 被拆分成2个函数执行
function BigThing() {
 doPartThing();

 process.nextTick(() => finishThing());
}

在事件循环中,何时执行 nextTick 注册的任务呢?请看下面的代码:

setTimeout(function() {
 console.log("第一个1秒");
 process.nextTick(function() {
  console.log("第一个1秒:nextTick");
 });
}, 1000);

setTimeout(function() {
 console.log("第2个1秒");
}, 1000);

console.log("我要输出1");

process.nextTick(function() {
 console.log("nextTick");
});

console.log("我要输出2");

输出的结果如下,nextTick 是早于 setTimeout:

我要输出1
我要输出2
nextTick
第一个1秒
第一个1秒:nextTick
第2个1秒

在浏览器端,nextTick 会退化成 setTimeout(callback, 0)。但在 nodejs 中请使用 nextTick 而不是 setTimeout,前者效率更高,并且严格来说,两者创建的事件在任务队列中顺序并不一样(请看前面的代码)。

以上就是详解Node.JS模块 process的详细内容,更多关于Node.JS模块 process的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
jquery radio 操作代码
Mar 16 Javascript
node.js中的querystring.stringify方法使用说明
Dec 10 Javascript
JQuery中$.each 和$(selector).each()的区别详解
Mar 13 Javascript
javascript获取网页宽高方法汇总
Jul 19 Javascript
JS实现横向与竖向两个选项卡Tab联动的方法
Sep 27 Javascript
使用three.js 画渐变的直线
Jun 05 Javascript
vue+vux实现移动端文件上传样式
Jul 28 Javascript
微信小程序swiper实现滑动放大缩小效果
Nov 15 Javascript
微信小程序Page中data数据操作和函数调用方法
May 08 Javascript
vue滚动插件better-scroll使用详解
Oct 18 Javascript
如何使用Jquery动态生成二级选项列表
Feb 06 jQuery
vue组件库的在线主题编辑器的实现思路
Apr 03 Javascript
谈谈JavaScript令人迷惑的==与+
Aug 31 #Javascript
vue自定义指令限制输入框输入值的步骤与完整代码
Aug 30 #Javascript
JS性能优化实现方法及优点进行
Aug 30 #Javascript
如何检测JavaScript中的死循环示例详解
Aug 30 #Javascript
JavaScript中CreateTextFile函数
Aug 30 #Javascript
详解vue组件之间的通信
Aug 30 #Javascript
如何阻止移动端浏览器点击图片浏览
Aug 29 #Javascript
You might like
PHP中创建空文件的代码[file_put_contents vs touch]
2012/01/20 PHP
Codeigniter校验ip地址的方法
2015/03/21 PHP
Linux系统下PHP-FPM的安装和配置教程
2015/08/17 PHP
PHP验证码无法显示的原因及解决办法
2017/08/11 PHP
JavaScript 基本概念
2015/01/20 Javascript
javascript中的正则表达式使用详解
2015/08/30 Javascript
BootStrap智能表单实战系列(八)表单配置json详解
2016/06/13 Javascript
JavaScript检测原始值、引用值、属性
2016/06/20 Javascript
js制作支付倒计时页面
2016/10/21 Javascript
深入理解jquery的$.extend()、$.fn和$.fn.extend()
2017/07/08 jQuery
Vue resource中的GET与POST请求的实例代码
2017/07/21 Javascript
什么是Vue.js框架 为什么选择它?
2017/10/17 Javascript
利用JavaScript的%做隔行换色的实例
2017/11/25 Javascript
javascript数组拍平方法总结
2018/01/20 Javascript
vue实现循环切换动画
2018/10/17 Javascript
浅谈Vue.js 中的 v-on 事件指令的使用
2018/11/25 Javascript
layui上传图片到服务器的非项目目录下的方法
2019/09/26 Javascript
浅谈vue中组件绑定事件时是否加.native
2019/11/09 Javascript
python在windows下实现ping操作并接收返回信息的方法
2015/03/20 Python
python自定义解析简单xml格式文件的方法
2015/05/11 Python
使用matplotlib画散点图的方法
2018/05/25 Python
Python交互环境下实现输入代码
2018/06/22 Python
解决python "No module named pip" 的问题
2018/10/13 Python
对sklearn的使用之数据集的拆分与训练详解(python3.6)
2018/12/14 Python
Python实现的IP端口扫描工具类示例
2019/02/15 Python
Python3解释器知识点总结
2019/02/19 Python
python批量修改文件夹及其子文件夹下的文件内容
2019/03/15 Python
python super用法及原理详解
2020/01/20 Python
使用PyCharm官方中文语言包汉化PyCharm
2020/11/18 Python
css3选择器基本介绍
2014/12/15 HTML / CSS
美国知名日用品连锁超市:Dollar General(多来店)
2017/01/14 全球购物
董事长秘书岗位职责
2013/11/29 职场文书
关于廉洁的广播稿
2014/01/30 职场文书
贺卡寄语大全
2014/04/11 职场文书
创业计划书之餐饮馄饨店
2019/07/18 职场文书
Python 数据结构之十大经典排序算法一文通关
2021/10/16 Python