穩(wěn)定性: 3 - 穩(wěn)定
Read-Eval-Print-Loop (REPL 讀取-執(zhí)行-輸出循環(huán))即可作為獨立程序,也可以集成到其他程序中。REPL 提供了一種交互的執(zhí)行 JavaScript 并查看輸出結(jié)果的方法。可以用來調(diào)試,測試,或僅是用來試試。
在命令行中不帶任何參數(shù)的執(zhí)行 node
,就是 REPL 模式。它提供了簡單的 emacs 行編輯。
mjr:~$ node
Type '.help' for options.
> a = [ 1, 2, 3];
[ 1, 2, 3 ]
> a.forEach(function (v) {
... console.log(v);
... });
1
2
3
若想使用高級的編輯模式,使用環(huán)境變量 NODE_NO_READLINE=1
打開 node。這樣會開啟 REPL 模式,允許你使用 rlwrap
。
例如,你可以添加以下代碼到你的 bashrc 文件里。
alias node="env NODE_NO_READLINE=1 rlwrap node"
啟動并返回一個 REPLServer
實例。它繼承自[Readline Interface][]。接收的參數(shù) "options" 有以下值:
prompt
- 所有輸入輸出的提示符和流。默認(rèn)是 >
.
input
- 需要監(jiān)聽的可讀流,默認(rèn) process.stdin
.
output
- 用來輸出數(shù)據(jù)的可寫流,默認(rèn)為 process.stdout
.
terminal
- 如果 stream
被當(dāng)成 TTY,并且有 ANSI/VT100 轉(zhuǎn)義, 傳true
。默認(rèn)在實例的輸出流上檢查isTTY
。
eval
- 用來對每一行進(jìn)行求值的函數(shù)。默認(rèn)為 eval()
的異步封裝。參見后面的自定義 eval
例子。
useColors
- 寫函數(shù)輸出是否有顏色。如果設(shè)定了不同的 writer
函數(shù)則無效。默認(rèn)為 repl 的 terminal
值。
useGlobal
- 如果為 true
,則 repl 將會使用全局對象,而不是在獨立的上下文中運行scripts。默認(rèn)為 false
。
ignoreUndefined
- 如果為 true
,repl 不會輸出未定義命令的返回值。默認(rèn)為 false
。
writer
- 每個命令行被求值時都會調(diào)用這個函數(shù),它會返回格式化顯示內(nèi)容(包括顏色)。默認(rèn)是 util.inspect
。如果有以下特性,可以使用自己的 eval
函數(shù):
function eval(cmd, context, filename, callback) {
callback(null, result);
}
在同一個 node 的運行實例上,可以打開多個 REPLs。每個都會共享一個全局對象,但會有獨立的 I/O。
以下的例子,在stdin, Unix socket, 和 TCP socket 上開啟 REPL :
var net = require("net"),
repl = require("repl");
connections = 0;
repl.start({
prompt: "node via stdin> ",
input: process.stdin,
output: process.stdout
});
net.createServer(function (socket) {
connections += 1;
repl.start({
prompt: "node via Unix socket> ",
input: socket,
output: socket
}).on('exit', function() {
socket.end();
})
}).listen("/tmp/node-repl-sock");
net.createServer(function (socket) {
connections += 1;
repl.start({
prompt: "node via TCP socket> ",
input: socket,
output: socket
}).on('exit', function() {
socket.end();
});
}).listen(5001);
從命令行運行這個程序,將會在 stdin 上啟動 REPL。其他的 REPL 客戶端可能通過 Unix socket 或 TCP socket 連接。telnet
常用于連接 TCP socket, socat
用于連接Unix 和 TCP sockets
從Unix socket-based 服務(wù)器啟動 REPL(而非stdin),你可以建立長連接,不用重啟它們。
通過 net.Server
和 net.Socket
實例運行"full-featured" (terminal
) REPL的例子,參見: https://gist.github.com/2209310
通過 curl(1)
實例運行 REPL的例子,參見: https://gist.github.com/2053342
function () {}
當(dāng)用戶通過預(yù)定義的方式退出 REPL 將會觸發(fā)這個事件。預(yù)定義的方式包括,在 repl 里輸入 .exit
,按 Ctrl+C 兩次來發(fā)送 SIGINT 信號,或者在 input
流上按 Ctrl+D 來發(fā)送"end"。
監(jiān)聽 exit
的例子:
r.on('exit', function () {
console.log('Got "exit" event from repl!');
process.exit();
});
function (context) {}
重置 REPL 的上下文的時候觸發(fā)。當(dāng)你輸入.clear
會重置。如果你用 { useGlobal: true }
啟動 repl,那這個事件永遠(yuǎn)不會被觸發(fā)。
監(jiān)聽 reset
的例子:
// Extend the initial repl context.
r = repl.start({ options ... });
someExtension.extend(r.context);
// When a new context is created extend it as well.
r.on('reset', function (context) {
console.log('repl has a new context');
someExtension.extend(context);
});
在 REPL 里, Control+D 會退出。可以輸入多行表達(dá)式。支持全局變量和局部變量的 TAB 自動補全。
特殊變量_
(下劃線)包含上一個表達(dá)式的結(jié)果。
> [ "a", "b", "c" ]
[ 'a', 'b', 'c' ]
> _.length
3
> _ += 1
4
REPL支持在全局域里訪問任何變量。將變量賦值個和REPLServer
關(guān)聯(lián)的上下文對象,你可以顯示的講變量暴露給 REPL,例如:
// repl_test.js
var repl = require("repl"),
msg = "message";
repl.start("> ").context.m = msg;
context
對象里的東西,會以局部變量的形式出現(xiàn):
mjr:~$ node repl_test.js
> m
'message'
有一些特殊的REPL命令:
.break
- 當(dāng)你輸入多行表達(dá)式時,也許你走神了或者不想完成了,.break
可以重新開始。 .clear
- 重置 context
對象為空對象,并且清空多行表達(dá)式。 .exit
- 關(guān)閉輸入/輸出流,會讓 REPL 退出。.help
- 打印這些特殊命令。.save
- 保存當(dāng)前 REPL 會話到文件。.save ./file/to/save.js
.load
- 加載一個文件到當(dāng)前REPL 會話.load ./file/to/load.js
下面的組合鍵在 REPL 中有以下效果:
<ctrl>C
- 和 .break
鍵類似. 在一個空行連按兩次會強制退出。<ctrl>D
- 和 .exit
鍵類似。