Process

最后更新于:2022-04-01 01:58:52

`process`对象是一个全局对象,并且何以被在任何地方调用。这是一个`EventEmitter`实例。 ### Exit Codes 当没有任何异步操作在等待时,`io.js`通常将会以一个0为退出码退出。以下这些状态码会在其他情况下被用到: - 1 未捕获的致命异常。这是一个未捕获的异常,并且它没有被`domain`处理,也没有被`uncaughtException`处理。 - 2 未使用(由`Bash`为内建误操作保留)。 - 3 内部的`JavaScript`解析错误。`io.js`内部的`JavaScript`源码引导(bootstrapping)造成的一个解释错误。这极其罕见。并且常常只会发生在`io.js`自身的开发过程中。 - 4 内部的`JavaScript`求值错误。`io.js`内部的`JavaScript`源码引导(bootstrapping)未能在求值时返回一个函数值。这极其罕见。并且常常只会发生在`io.js`自身的开发过程中。 - 5 致命错误。这是V8中严重的不可恢复的错误。典型情况下,一个带有`FATAL ERROR`前缀的信息会被打印在`stderr`。 - 6 内部异常处理函数丧失功能。这是一个未捕获异常,但是内部的致命异常处理函数被设置为丧失功能,并且不能被调用。 - 7 内部异常处理函数运行时失败。这是一个未捕获异常,并且内部致命异常处理函数试图处理它时,自身抛出了一个错误。例如它可能在当`process.on('uncaughtException')`或`domain.on('error')`处理函数抛出错误时发生。 - 8 未使用。`io.js`的之前版本中,退出码`8`通常表示一个未捕获异常。 - 9 无效参数。当一个位置的选项被指定,或者一个必选的值没有被提供。 - 10 内部的`JavaScript`运行时错误。`io.js`内部的`JavaScript`源码引导(bootstrapping)函数被调用时抛出一个错误。这极其罕见。并且常常只会发生在`io.js`自身的开发过程中。 - 12 无效的调试参数。`--debug`和/或`--debug-brk`选项被设置,当时选择了一个无效的端口。 - 大于128 信号退出。如果`io.js`收到了一个如`SIGKILL`或`SIGHUP`的致命信号,那么它将以一个`128`加上 信号码的值 的退出码退出。这是一个标准的Unix实践,因为退出码由一个7位整数定义,并且信号的退出设置了一个高顺序位(high-order bit),然后包含一个信号码的值。 #### Event: 'exit' 进程即将退出时触发。在这个时刻已经没有办法可以阻止事件循环的退出,并且一旦所有的`exit`监听器运行结束时,进程将会退出。因此,在这个监听器中你仅仅能调用同步的操作。这是检查模块状态(如单元测试)的好钩子。回调函数有一个退出码参数。 例子: ~~~ process.on('exit', function(code) { // do *NOT* do this setTimeout(function() { console.log('This will not run'); }, 0); console.log('About to exit with code:', code); }); ~~~ #### Event: 'beforeExit' 这个事件在`io.js`清空了它的事件循环并且没有任何已安排的任务时触发。通常`io.js`当没有更多被安排的任务时就会退出,但是`beforeExit`中可以执行异步调用,让`io.js`继续运行。 `beforeExit`在程序被显示终止时不会触发,如`process.exit()`或未捕获的异常。除非想去安排更多的任务,否则它不应被用来做为`exit`事件的替代。 #### Event: 'uncaughtException' 当一个异常冒泡回事件循环时就会触发。如果这个时间被添加了监听器,那么默认行为(退出程序且打印堆栈跟踪信息)将不会发生。 例子: ~~~ process.on('uncaughtException', function(err) { console.log('Caught exception: ' + err); }); setTimeout(function() { console.log('This will still run.'); }, 500); // Intentionally cause an exception, but don't catch it. nonexistentFunc(); console.log('This will not run.'); ~~~ 注意,`uncaughtException`来处理异常是非常粗糙的。 请不要使用它,使用`domain`来替代。如果你已经使用了它,请在不处理这个异常之后重启你的应用。 请不要像`io.js`的`Error Resume Next`这样使用。一个未捕获异常意味着你的应用或拓展有未定义的状态。盲目地恢复意味着任何事都可能发生。 想象你在升级你的系统时电源被拉断了。10次中前9次都没有问题,但是第10次时,你的系统崩溃了。 你已经被警告。 #### Event: 'unhandledRejection' 在一个事件循环中,当一个`promise`被“拒绝”并且没有附属的错误处理函数时触发。当一个带有`promise`异常的程序被封装为被“拒绝”的`promise`时,这样的程序的错误可以被`promise.catch(...)`捕获处理并且“拒绝”会通过`promise`链冒泡。这个事件对于侦测和保持追踪那些“拒绝”没有被处理的`promise`非常有用。这个事件会带着以下参数触发: - reason `promise`的“拒绝”对象(通常是一个错误实例) - p 被“拒绝”的`promise` 下面是一个把所有未处理的“拒绝”打印到控制台的例子: ~~~ process.on('unhandledRejection', function(reason, p) { console.log("Unhandled Rejection at: Promise ", p, " reason: ", reason); // application specific logging, throwing an error, or other logic here }); ~~~ 下面是一个会触发`unhandledRejection`事件的“拒绝”: ~~~ somePromise.then(function(res) { return reportToUser(JSON.pasre(res)); // note the typo }); // no `.catch` or `.then` ~~~ #### Event: 'rejectionHandled' 当一个`Promise`被“拒绝”并且一个错误处理函数被附给了它(如`.catch()`)时的下一个事件循环之后触发。这个事件会带着以下参数触发: - p 一个在之前会被触发在`unhandledRejection`事件中,但现在被处理函数捕获的`promise` 一个`promise`链的顶端没有 “拒绝”可以总是被处理 的概念。由于其异步的本质,一个`promise`的“拒绝”可以在未来的某一个时间点被处理,可以是在事件循环中被触发`unhandledRejection`事件之后。 另外,不像同步代码中是一个永远增长的 未捕获异常 列表,`promise`中它是一个可伸缩的 未捕获`拒绝` 列表。在同步代码中,`uncaughtException`事件告诉你 未捕获异常 列表增长了。但是在`promise`中,`unhandledRejection`事件告诉你 未捕获“拒绝” 列表增长了,`rejectionHandled`事件告诉你 未捕获“拒绝” 列表缩短了。 使用“拒绝”侦测钩子来保持一个被“拒绝”的`promise`列表: ~~~ var unhandledRejections = []; process.on('unhandledRejection', function(reason, p) { unhandledRejections.push(p); }); process.on('rejectionHandled', function(p) { var index = unhandledRejections.indexOf(p); unhandledRejections.splice(index, 1); }); ~~~ #### Signal Events 当一个进程收到一个信号时触发。参阅`sigaction(2)`。 监听`SIGINT`信号的例子: ~~~ // Start reading from stdin so we don't exit. process.stdin.resume(); process.on('SIGINT', function() { console.log('Got SIGINT. Press Control-D to exit.'); }); ~~~ 一个发送`SIGINT`信号的快捷方法是在大多数终端中按下 Control-C 。 注意: - SIGUSR1 是`io.js`用于开启调试的保留信号。可以为其添加一个监听器,但不能阻止调试的开始。 - SIGTERM 和 SIGINT在非Windows平台下有在以 128 + 信号 退出码退出前重置终端模式的默认监听器。如果另有监听器被添加,默认监听器会被移除(即`io.js`将会不再退出)。 - SIGPIPE 默认被忽略,可以被添加监听器。 - SIGHUP 当控制台被关闭时会在Windows中产生,或者其他平台有其他相似情况时(参阅`signal(7)`)。它可以被添加监听器,但是Windows中`io.js`会无条件的在10秒后关闭终端。在其他非Windows平台,它的默认行为是结束`io.js`,但是一旦被添加了监听器,默认行为会被移除。 - SIGTERM 在Windows中不被支持,它可以被监听。 - SIGINT 支持所有的平台。可以由 CTRL+C 产生(尽管它可能是可配置的)。当启用终端的`raw mode`时,它不会产生。 - SIGBREAK 在Windows中,按下 CTRL+BREAK 时它会产生。在非Windows平台下,它可以被监听,但它没有产生的途径。 - SIGWINCH 当终端被改变大小时产生。Windows下,它只会在当光标被移动时写入控制台或可读tty使用`raw mode`时发生。 - SIGKILL 可以被添加监听器。它会无条件得在所有平台下关闭`io.js`。 - SIGSTOP 可以被添加监听器。 注意Windows不支持发送信号,但`io.js`通过`process.kill()`和`child_process.kill()`提供了模拟:- 发送信号`0`被用来检查进程的存在 - 发送SIGINT, SIGTERM 和 SIGKILL 会导致目标进程的无条件退出。 #### process.stdout 一个指向`stdout`的可写流。 例如,`console.log`可能与这个相似: ~~~ console.log = function(msg) { process.stdout.write(msg + '\n'); }; ~~~ 在`io.js`中,`process.stderr`和`process.stdout`与其他流不同,因为他们不能被关闭(调用`end()`会报错)。它们永远不触发`finish`事件并且写操作通常是阻塞的。 - 当指向普通文件或TTY文件描述符时,它们是阻塞的。 - **以下情况下他们指向流** - 他们在Linux/Unix中阻塞 - 他们在Windows中的其他流里不阻塞 若要检查`io.js`是否在一个TTY上下文中运行,读取`process.stderr`,`process.stdout`或`process.stdin`的`isTTY`属性: ~~~ $ iojs -p "Boolean(process.stdin.isTTY)" true $ echo "foo" | iojs -p "Boolean(process.stdin.isTTY)" false $ iojs -p "Boolean(process.stdout.isTTY)" true $ iojs -p "Boolean(process.stdout.isTTY)" | cat false ~~~ 更多信息请参阅tty文档。 #### process.stderr 一个指向`stderr`的可写流。 在`io.js`中,`process.stderr`和`process.stdout`与其他流不同,因为他们不能被关闭(调用`end()`会报错)。它们永远不触发`finish`事件并且写操作通常是阻塞的。 - 当指向普通文件或TTY文件描述符时,它们是阻塞的。 - **以下情况下他们指向流** - 他们在Linux/Unix中阻塞 - 他们在Windows中的其他流里不阻塞 #### process.stdin 一个指向`stdin`的可读流。 一个打开标准输入并且监听两个事件的例子: ~~~ process.stdin.setEncoding('utf8'); process.stdin.on('readable', function() { var chunk = process.stdin.read(); if (chunk !== null) { process.stdout.write('data: ' + chunk); } }); process.stdin.on('end', function() { process.stdout.write('end'); }); ~~~ 作为一个流,`process.stdin`可以被切换至“旧”模式,这样就可以兼容`node.js` v0.10 前所写的脚本。更多信息请参阅 流的兼容性 。 在“旧”模式中`stdin`流默认是被暂停的。所以你必须调用`process.stdin.resume()`来读取。注意调用`process.stdin.resume()`这个操作本身也会将流切换至旧模式。 如果你正将开启一个新的工程。你应该要更常使用“新”模式的流。 #### process.argv 一个包含了命令行参数的数组。第一次元素将会是`'iojs'`,第二个元素将会是`JavaScript`文件名。之后的元素将会是额外的命令行参数。 ~~~ // print process.argv process.argv.forEach(function(val, index, array) { console.log(index + ': ' + val); }); ~~~ 这将会是: ~~~ $ iojs process-2.js one two=three four 0: iojs 1: /Users/mjr/work/iojs/process-2.js 2: one 3: two=three 4: four ~~~ #### process.execPath 这将是开启进程的可执行文件的绝对路径名: 例子: ~~~ /usr/local/bin/iojs ~~~ #### process.execArgv 这是在启动时`io.js`自身参数的集合。这些参数不会出现在`process.argv`中,并且不会包含`io.js`可执行文件,脚本名和其他脚本名之后的参数。这些参数对开启和父进程相同执行环境的子进程非常有用。 例子: ~~~ $ iojs --harmony script.js --version ~~~ `process.execArgv`将会是: ~~~ ['--harmony'] ~~~ `process.argv`将会是: ~~~ ['/usr/local/bin/iojs', 'script.js', '--version'] ~~~ #### process.abort() 这将导致`io.js`触发`abort`事件。这个将导致`io.js`退出,并创建一个核心文件。 #### process.chdir(directory) 为进程改变当前工作目录,如果失败,则抛出一个异常。 ~~~ console.log('Starting directory: ' + process.cwd()); try { process.chdir('/tmp'); console.log('New directory: ' + process.cwd()); } catch (err) { console.log('chdir: ' + err); } ~~~ #### process.cwd() 返回进程的当前工作目录。 ~~~ console.log('Current directory: ' + process.cwd()); ~~~ #### process.env 包含用户环境变量的对象。参阅`environ(7)`。 一个例子: ~~~ { TERM: 'xterm-256color', SHELL: '/usr/local/bin/bash', USER: 'maciej', PATH: '~/.bin/:/usr/bin:/bin:/usr/sbin:/sbin:/usr/local/bin', PWD: '/Users/maciej', EDITOR: 'vim', SHLVL: '1', HOME: '/Users/maciej', LOGNAME: 'maciej', _: '/usr/local/bin/iojs' } ~~~ 你可以改写这个对象,但是改变不会反应在你的进程之外。这以为着以下代码不会正常工作: ~~~ $ iojs -e 'process.env.foo = "bar"' && echo $foo ~~~ 但是以下代码会: ~~~ process.env.foo = 'bar'; console.log(process.env.foo); ~~~ #### process.exit([code]) 使用指定的退出码退出程序,如果忽略退出码。那么将使用“成功”退出码`0`。 以一个“失败”退出码结束: ~~~ process.exit(1); ~~~ 在执行`io.js`的shell中可以看到为`1`的退出码。 #### process.exitCode 将是程序退出码的数字,当程序优雅退出 或 被`process.exit()`关闭且没有指定退出码时。 为`process.exit(code)`指定一个退出码会覆盖之前的`process.exitCode`设置。 #### process.getgid() 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 获取进程的群组标识(参阅`getgid(2)`)。这是一个群组id数组,不是群组名。 ~~~ if (process.getgid) { console.log('Current gid: ' + process.getgid()); } ~~~ #### process.getegid() 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 获取进程的有效群组标识(参阅`getgid(2)`)。这是一个群组id数组,不是群组名。 ~~~ if (process.getegid) { console.log('Current gid: ' + process.getegid()); } ~~~ #### process.setgid(id) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 设置进程的群组标识(参阅`setgid(2)`)。它接受一个数字ID或一个群组名字符串。如果群组名被指定,那么这个方法将在解析群组名为一个ID的过程中阻塞。 ~~~ if (process.getgid && process.setgid) { console.log('Current gid: ' + process.getgid()); try { process.setgid(501); console.log('New gid: ' + process.getgid()); } catch (err) { console.log('Failed to set gid: ' + err); } } ~~~ #### process.setegid(id) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 设置进程的有效群组标识(参阅`setgid(2)`)。它接受一个数字ID或一个群组名字符串。如果群组名被指定,那么这个方法将在解析群组名为一个ID的过程中阻塞。 ~~~ if (process.getegid && process.setegid) { console.log('Current gid: ' + process.getegid()); try { process.setegid(501); console.log('New gid: ' + process.getegid()); } catch (err) { console.log('Failed to set gid: ' + err); } } ~~~ #### process.getuid() 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 获取进程的用户id(参阅`getuid(2)`)。这是一个数字用户id,不是用户名。 ~~~ if (process.getuid) { console.log('Current uid: ' + process.getuid()); } ~~~ #### process.geteuid() 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 获取进程的有效用户id(参阅`getuid(2)`)。这是一个数字用户id,不是用户名。 ~~~ if (process.geteuid) { console.log('Current uid: ' + process.geteuid()); } ~~~ #### process.setuid(id) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 设置进程的用户ID(参阅`setuid(2)`)。它接受一个数字ID或一个用户名字符串。如果用户名被指定,那么这个方法将在解析用户名为一个ID的过程中阻塞。 ~~~ if (process.getuid && process.setuid) { console.log('Current uid: ' + process.getuid()); try { process.setuid(501); console.log('New uid: ' + process.getuid()); } catch (err) { console.log('Failed to set uid: ' + err); } } ~~~ #### process.seteuid(id) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 设置进程的有效用户ID(参阅`seteuid(2)`)。它接受一个数字ID或一个用户名字符串。如果用户名被指定,那么这个方法将在解析用户名为一个ID的过程中阻塞。 ~~~ if (process.geteuid && process.seteuid) { console.log('Current uid: ' + process.geteuid()); try { process.seteuid(501); console.log('New uid: ' + process.geteuid()); } catch (err) { console.log('Failed to set uid: ' + err); } } ~~~ #### process.getgroups() 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 返回一个补充群组ID的数组。如果包含了有效的组ID,POSIX将不会指定。但`io.js`保证它始终是。 #### process.setgroups(groups) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 设置一个补充群组ID。这是一个特殊的操作,意味着你需要拥有`root`或`CAP_SETGID`权限才可以这么做。 列表可以包含群组ID,群组名,或两者。 #### process.initgroups(user, extra_group) 注意:这个函数只在POSIX平台上有效(如在Windows,Android中无效)。 读取`/etc/group`并且初始化群组访问列表,使用用户是组员的所有群组。这是一个特殊的操作,意味着你需要拥有`root`或`CAP_SETGID`权限才可以这么做。 `user`是一个用户名或一个用户ID。`extra_group`是一个群组名或群组ID。 当你注销权限时有些需要关心的: ~~~ console.log(process.getgroups()); // [ 0 ] process.initgroups('bnoordhuis', 1000); // switch user console.log(process.getgroups()); // [ 27, 30, 46, 1000, 0 ] process.setgid(1000); // drop root gid console.log(process.getgroups()); // [ 27, 30, 46, 1000 ] ~~~ #### process.version 一个暴露`NODE_VERSION`的编译时存储属性。 ~~~ console.log('Version: ' + process.version); ~~~ #### process.versions 一个暴露io.js版本和它的依赖的字符串属性。 ~~~ console.log(process.versions); ~~~ 将可能打印: ~~~ { http_parser: '2.3.0', node: '1.1.1', v8: '4.1.0.14', uv: '1.3.0', zlib: '1.2.8', ares: '1.10.0-DEV', modules: '43', openssl: '1.0.1k' } ~~~ #### process.config 一个表示用于编译当前`io.js`执行文件的配置的JavaScript对象。这和运行`./configure`脚本产生的`config.gypi`一样。 一个可能的输出: ~~~ { target_defaults: { cflags: [], default_configuration: 'Release', defines: [], include_dirs: [], libraries: [] }, variables: { host_arch: 'x64', node_install_npm: 'true', node_prefix: '', node_shared_cares: 'false', node_shared_http_parser: 'false', node_shared_libuv: 'false', node_shared_zlib: 'false', node_use_dtrace: 'false', node_use_openssl: 'true', node_shared_openssl: 'false', strict_aliasing: 'true', target_arch: 'x64', v8_use_snapshot: 'true' } } ~~~ #### process.kill(pid[, signal]) 给进程传递一个信号。`pid`是进程id,`signal`是描述信号的字符串。信号码类似于`'SIGINT'`或`'SIGHUP'`。如果忽略,那么信号将是`'SIGTERM'`。更多信息参阅`Signal Events`和`kill(2)`。 如果目标不存在将会抛出一个错误,并且在一些情况下,`0`信号可以被用来测试进程的存在。 注意,这个函数仅仅是名字为`process.kill`,它只是一个信号发送者。发送的信号可能与杀死进程无关。 一个发送信号给自身的例子: ~~~ process.on('SIGHUP', function() { console.log('Got SIGHUP signal.'); }); setTimeout(function() { console.log('Exiting.'); process.exit(0); }, 100); ~~~ process.kill(process.pid, 'SIGHUP'); 注意:当`SIGUSR1`被`io.js`收到,它会开始调试。参阅`Signal Events`。 process.pid# 进程的PID。 ~~~ console.log('This process is pid ' + process.pid); ~~~ process.title# 设置/获取 `'ps'` 中显示的进程名。 当设置该属性时,所能设置的字符串最大长度视具体平台而定,如果超过的话会自动截断。 在 Linux 和 OS X 上,它受限于名称的字节长度加上命令行参数的长度,因为它有覆盖参数内存。 v0.8 版本允许更长的进程标题字符串,也支持覆盖环境内存,但是存在潜在的不安全和混乱。 #### process.arch 返回当前的处理器结构:`'arm'`,`'ia32'`或`'x64'`。 ~~~ console.log('This processor architecture is ' + process.arch); ~~~ ### process.platform 放回当前的平台:`'darwin'`,`'freebsd'`,`'linux'`,`'sunos'`或`'win32'`。 ~~~ console.log('This platform is ' + process.platform); ~~~ #### process.memoryUsage() 返回当前`io.js`进程内存使用情况(用字节描述)的对象。 ~~~ var util = require('util'); console.log(util.inspect(process.memoryUsage())); ~~~ 可能的输出: ~~~ { rss: 4935680, heapTotal: 1826816, heapUsed: 650472 } ~~~ `heapTotal`和`heapUsed`指向V8的内存使用。 #### process.nextTick(callback[, arg][, ...]) - callback Function 在事件循环的下一次循环中调用回调函数。 这不是`setTimeout(fn, 0)`的简单别名,它更有效率。在之后的`tick`中,它在任何其他的I/O事件(包括`timer`)触发之前运行。 ~~~ console.log('start'); process.nextTick(function() { console.log('nextTick callback'); }); console.log('scheduled'); // Output: // start // scheduled // nextTick callback ~~~ 这对于开发你想要给予用户在对象被构建后,任何I/O发生前,去设置事件监听器的机会时,非常有用。 ~~~ function MyThing(options) { this.setupOptions(options); process.nextTick(function() { this.startDoingStuff(); }.bind(this)); } var thing = new MyThing(); thing.getReadyForStuff(); // thing.startDoingStuff() gets called now, not before. ~~~ 这对于100%同步或100%异步的API非常重要。考虑一下例子: ~~~ // WARNING! DO NOT USE! BAD UNSAFE HAZARD! function maybeSync(arg, cb) { if (arg) { cb(); return; } fs.stat('file', cb); } ~~~ 这个API是危险的,如果你这样做: ~~~ maybeSync(true, function() { foo(); }); bar(); ~~~ `foo()`和`bar()`的调用次序是不确定的。 更好的做法是: ~~~ function definitelyAsync(arg, cb) { if (arg) { process.nextTick(cb); return; } fs.stat('file', cb); } ~~~ 注意:`nextTick`队列在每一次事件循环的I/O开始前都要完全执行完毕。所以,递归地设置`nextTick`回调会阻塞I/O的方法,就像一个`while(true);`循环。 #### process.umask([mask]) 设置或读取进程的文件模式的创建掩码。子进程从父进程中继承这个掩码。返回旧的掩码如果`mask`参数被指定。否则,会返回当前掩码。 ~~~ var oldmask, newmask = 0022; oldmask = process.umask(newmask); console.log('Changed umask from: ' + oldmask.toString(8) + ' to ' + newmask.toString(8)); ~~~ #### process.uptime() `io.js`进程已执行的秒数。 #### process.hrtime() 以`[seconds, nanoseconds]`元组数组的形式返回高分辨时间。是相对于过去的任意时间。它与日期无关所以不用考虑时区等因素。它的主要用途是衡量程序性能。 你可以将之前的`process.hrtime()`返回传递给一个新的`process.hrtime()`来获得一个比较。衡量性能时非常有用: ~~~ var time = process.hrtime(); // [ 1800216, 25 ] setTimeout(function() { var diff = process.hrtime(time); // [ 1, 552 ] console.log('benchmark took %d nanoseconds', diff[0] * 1e9 + diff[1]); // benchmark took 1000000527 nanoseconds }, 1000); ~~~ #### process.mainModule 检索`require.main`的备用方式。区别是,如果主模块在运行时改变,`require.main`可能仍指向改变发生前的被引入的原主模块。通常,假设它们一样是安全的。 与`require.main`一样,当如果没有入口脚本时,它将是`undefined`。
';