资讯专栏INFORMATION COLUMN

Node.js process模块解读

Riddler / 2814人阅读

摘要:默认情况下,会打印堆栈信息到然后退出进程。适用于父子进程之间发送消息,关于如何创建父子进程会放在模块中进行。信号虽然也是用于请求终止进程,但是它与有所不同,进程可以选择响应还是忽略此信号。

process存在于全局对象上,不需要使用require()加载即可使用,process模块主要做两方面的事情

读:获取进程信息(资源使用、运行环境、运行状态)

写:执行进程操作(监听事件、调度任务、发出警告)

资源使用

资源使用指运行此进程所消耗的机器资源。例如内存、cpu

内存
process.memoryUsage())

{ rss: 21848064,
  heapTotal: 7159808,
  heapUsed: 4431688,
  external: 8224 
 }

rss(常驻内存)的组成见下图

code segment对应当前运行的代码

external对应的是C++对象(与V8管理的JS对象绑定)的占用的内存,比如Buffer的使用

Buffer.allocUnsafe(1024 * 1024 * 1000);
console.log(process.memoryUsage());

{ rss: 22052864,
  heapTotal: 6635520,
  heapUsed: 4161376,
  external: 1048584224 }
cpu
const startUsage = process.cpuUsage();
console.log(startUsage);

const now = Date.now();
while (Date.now() - now < 500);

console.log(process.cpuUsage());
console.log(process.cpuUsage(startUsage)); //相对时间

// { user: 59459, system: 18966 }
// { user: 558135, system: 22312 }
// { user: 498432, system: 3333 }

user对应用户时间,system代表系统时间

运行环境

运行环境指此进程运行的宿主环境包括运行目录、node环境、CPU架构、用户环境、系统平台

运行目录
console.log(`Current directory: ${process.cwd()}`);

// Current directory: /Users/xxxx/workspace/learn/node-basic/process
node环境
console.log(process.version)

// v9.1.0

如果不仅仅希望获得node的版本信息,还希望v8、zlib、libuv版本等信息的话就需要使用process.versions了

console.log(process.versions);
{ http_parser: "2.7.0",
  node: "9.1.0",
  v8: "6.2.414.32-node.8",
  uv: "1.15.0",
  zlib: "1.2.11",
  ares: "1.13.0",
  modules: "59",
  nghttp2: "1.25.0",
  openssl: "1.0.2m",
  icu: "59.1",
  unicode: "9.0",
  cldr: "31.0.1",
  tz: "2017b" }
cpu架构
console.log(`This processor architecture is ${process.arch}`);

// This processor architecture is x64

支持的值包括:"arm", "arm64", "ia32", "mips", "mipsel", "ppc", "ppc64", "s390", "s390x", "x32" "x64"

用户环境
console.log(process.env.NODE_ENV); // dev

NODE_ENV=dev node b.js

除了启动时的自定义信息之外,process.env还可以获得其他的用户环境信息(比如PATH、SHELL、HOME等),感兴趣的可以自己打印一下试试

系统平台
console.log(`This platform is ${process.platform}`);

This platform is darwin

支持的系统平台包括:"aix" "darwin" "freebsd" "linux" "openbsd" "sunos" "win32"

android目前还处于试验阶段

运行状态

运行状态指当前进程的运行相关的信息包括启动参数、执行目录、主文件、PID信息、运行时间

启动参数

获取启动参数有三个方法,execArgv获取Node.js的命令行选项(见官网文档)

argv获取非命令行选项的信息,argv0则获取argv[0]的值(略有差异)

console.log(process.argv)
console.log(process.argv0)
console.log(process.execArgv)

node --harmony  b.js foo=bar --version

// 输出结果
[ "/Users/xiji/.nvm/versions/node/v9.1.0/bin/node",
  "/Users/xiji/workspace/learn/node-basic/process/b.js",
  "foo=bar",
  "--version" ]
node
[ "--harmony" ]
执行目录
console.log(process.execPath);

// /Users/xxxx/.nvm/versions/node/v9.1.0/bin/node
运行时间
var date = new Date();
while(new Date() - date < 500) {}
console.log(process.uptime()); // 0.569
主文件

除了require.main之外也可以通过process.mainModule来判断一个模块是否是主文件

//a.js
console.log(`module A: ${process.mainModule === module}`);

//b.js
require("./a");
console.log(`module B: ${process.mainModule === module}`);

node b.js
// 输出
module A: false
module B: true

PID信息

console.log(`This process is pid ${process.pid}`); //This process is pid 12554
监听事件

process是EventEmiiter的实例对象,因此可以使用process.on("eventName", () => {})来监听事件。
常用的事件类型分两种:

进程状态 比如:beforeExit、exit、uncaughtException、message

信号事件 比如:SIGTERM、SIGKILL、SIGUSR1

beforeExit与exit的区别有两方面:

beforeExit里面可以执行异步代码、exit只能是同步代码

手动调用process.exit()或者触发uncaptException导致进程退出不会触发beforeExit事件、exit事件会触发。

因此下面的代码console都不会被执行

process.on("beforeExit", function(code) {
  console.log("before exit: "+ code);
});
process.on("exit", function(code) {
  setTimeout(function() {
    console.log("exit: " + code);
  }, 0);
});
a.b();

当异常一直没有被捕获处理的话,最后就会触发"uncaughtException"事件。默认情况下,Node.js会打印堆栈信息到stderr然后退出进程。不要试图阻止uncaughtException退出进程,因此此时程序的状态可能已经不稳定了,建议的方式是及时捕获处理代码中的错误,uncaughtException里面只做一些清理工作。

注意:node的9.3版本增加了process.setUncaughtExceptionCaptureCallback方法

当process.setUncaughtExceptionCaptureCallback(fn)指定了监听函数的时候,uncaughtException事件将会不再被触发。

process.on("uncaughtException", function() {
  console.log("uncaught listener");
});

process.setUncaughtExceptionCaptureCallback(function() {
  console.log("uncaught fn");
});

a.b();
// uncaught fn

message适用于父子进程之间发送消息,关于如何创建父子进程会放在child_process模块中进行。

SIGTERM信号虽然也是用于请求终止Node.js进程,但是它与SIGKILL有所不同,进程可以选择响应还是忽略此信号。
SIGTERM会以一种友好的方式来结束进程,在进程结束之前先释放已分配的资源(比如数据库连接),因此这种方式被称为优雅关闭(graceful shutdown)
具体的执行步骤如下:

应用程序被通知需要关闭(接收到SIGTERM信号)

应用程序通知负载均衡不再接收新的请求

应用程序完成正在进行中的请求

释放资源(例如数据库连接)

应用程序正常退出,退出状态码为0

SIGUSR1
Node.js当接收到SIGUSR1信号时会启动内置的调试器,当执行下列操作时

kill -USR1 PID_OF_THE_NODE_JS_PROCESS

可以看到node.js会启动调试器代理,端口是9229

server is listening 8089
Debugger listening on ws://127.0.0.1:9229/7ef98ccb-02fa-451a-8954-4706bd74105f
For help, see: https://nodejs.org/en/docs/inspector

也可以在服务启动时使用--inspect 来启动调试代理

node --inspect index.js
调度任务

process.nextTick(fn)

通过process.nextTick调度的任务是异步任务,EventLoop是分阶段的,每个阶段执行特定的任务,而nextTick的任务在阶段切换的时候就会执行,因此nextTick会比setTimeout(fn, 0)更快的执行,关于EventLoop见下图,后面会做进一步详细的讲解

发出警告
process.emitWarning("Something warning happened!", {
  code: "MY_WARNING",
  type: "XXXX"
});

// (node:14771) [MY_WARNING] XXXX: Something warning happened!

当type为DeprecationWarning时,可以通过命令行选项施加影响

--throw-deprecation 会抛出异常

--no-deprecation 不输出DeprecationWarning

--trace-deprecation 打印详细堆栈信息

process.emitWarning("Something warning happened!", {
  type: "DeprecationWarning"
});
console.log(4);

node --throw-deprecation index.js
node --no-deprecation index.js
node --trace-deprecation index.js

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/107838.html

相关文章

  • Node.js child_process模块解读

    摘要:而且方式创建的子进程与父进程之间建立了通信管道,因此子进程和父进程之间可以通过的方式发送消息。与事件的回调函数有两个参数和,代码子进程最终的退出码,如果子进程是由于接收到信号终止的话,会记录子进程接受的值。 在介绍child_process模块之前,先来看一个下面的代码。 const http = require(http); const longComputation = () =>...

    baiy 评论0 收藏0
  • 关于Node.js的__dirname,__filename,process.cwd(),./文件路

    摘要:先把当成文件,依次查找当前目录下的,找到了,就返回该文件,不再继续执行。那么关于正确的结论是在中使用是跟的效果相同,不会因为启动脚本的目录不一样而改变,在其他情况下跟效果相同,是相对于启动脚本所在目录的路径。 起因 原文收录在我的 GitHub博客 (https://github.com/jawil/blog) ,喜欢的可以关注最新动态,大家一起多交流学习,共同进步,以学习者的身份写博...

    harryhappy 评论0 收藏0
  • 关于Node.js的__dirname,__filename,process.cwd(),./文件路

    摘要:先把当成文件,依次查找当前目录下的,找到了,就返回该文件,不再继续执行。那么关于正确的结论是在中使用是跟的效果相同,不会因为启动脚本的目录不一样而改变,在其他情况下跟效果相同,是相对于启动脚本所在目录的路径。 起因 原文收录在我的 GitHub博客 (https://github.com/jawil/blog) ,喜欢的可以关注最新动态,大家一起多交流学习,共同进步,以学习者的身份写博...

    pumpkin9 评论0 收藏0
  • 如何打造一个令人愉悦的前端开发环境(四)

    摘要:在前后端分离的前端项目开发中经常用到。是的一个中间件。即是一个重要的功能。配置先来在配置文件中引入添加一个和通信的客户端添加应用入口文件在插件中引入在我们的开发环境中是这样配置的。 原文链接此文是我同事写的,搭建Express结合Webpack。以下是正文,后面我会附上我的解读 Express 结合 Webpack 实现HMR 本篇文件主要讲结合 Webpack 和 Express 实...

    StonePanda 评论0 收藏0

发表评论

0条评论

Riddler

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<