言語仕様を拡張する
– 文字列
– 整数123, 実数123.4
– print関数でカンマ区切りで複数の引数指定
– 複数文は、セミコロンで区切る
– // から行末まではコメント
### source.3
print("引数1個目","引数1個目","引数1個目");
print("整数=",1234);
print("実数=",1234.5);
### lexer.js
\/\/.*$ =>コメント削除、mオプションが必要
\d+\.\d+ =>実数
\d+ => 整数
“.*?” => 文字列
\w+ => print
\s => 空白、改行
. => セミコロン、カッコ、その他
module.exports = function(source){
var tokens = source.split(/\/\/.*$|(\d+\.\d+|\d+|".*?"|\w+)|\s|(.)/m);
tokens = tokens.filter(a=>a);
return tokens;
}
### parser.js
module.exports = parser;
var {expect, accept, show, error} = require("./utils.js");
var tokens;
function parser(t){
tokens = t;
var ast = semi();
if(tokens.length>0){
show("ast=", ast);
show("処理後tokens =", tokens);
error("tokensが余っています。");
}
return ast;
}
function value(){
if(tokens.length == 0) return;
return tokens.shift();
}
function funccall(){
var left = value();
var op;
while(op = accept(tokens, "(")){
var right = comma();
op += expect(tokens, ")");
left = {left, op, right};
}
return left;
}
function comma(){
var left = funccall();
var op;
while(op = accept(tokens,",")){
var right = funccall();
left = {left, op, right};
}
return left;
}
function semi(){
var left = comma();
var op;
while(op = accept(tokens, ";")){
var right = funccall();
left = {left, op, right};
}
return left;
}
### run.js
1234, 1234.5は文字列になっているので、1を掛けて数値化
tree形式から配列化している
flatに[left, right]に戻す
const { error } = require("./utils");
module.exports = run;
function run(a){
if(!a) return;
if(!a.op){
if(a[0] == '"') return a.substr(1, a.length-2);
if(/\d/.test(a[0])) return 1*a;
return a;
} else if (a.op==";"){
run(a.left);
run(a.right);
} else if (a.op==","){
return [run(a.left), run(a.right)].flat();
} else if(a.op == "()") {
var func = run(a.left);
if(func == "print"){
var msg = [run(a.right)].flat().join("");
console.log(msg);
} else {
error("未実装の関数呼び出し func=", func);
}
} else {
error("未実装の演算子 op=", a.op)
}
}
opに追加する場合は、字句解析の正規表現と実行時の処理を追加する
カンマがあった場合は、left, op, rightで入れ子にするが、なかった場合は、そのまま処理を続ける
$ node interpretor.js
token =[
‘print’, ‘(‘, ‘”引数1個目”‘,
‘,’, ‘”引数2個目”‘, ‘,’,
‘”引数3個目”‘, ‘)’, ‘;’,
‘print’, ‘(‘, ‘”引数4″‘,
‘)’, ‘;’, ‘print’,
‘(‘, ‘”整数=”‘, ‘,’,
‘1234’, ‘)’, ‘;’,
‘print’, ‘(‘, ‘”実数=”‘,
‘,’, ‘1234.5’, ‘)’,
‘;’
]
ast={
left: {
left: {
left: {
left: {
left: ‘print’,
op: ‘()’,
right: {
left: { left: ‘”引数1個目”‘, op: ‘,’, right: ‘”引数2個目”‘ },
op: ‘,’,
right: ‘”引数3個目”‘
}
},
op: ‘;’,
right: { left: ‘print’, op: ‘()’, right: ‘”引数4″‘ }
},
op: ‘;’,
right: {
left: ‘print’,
op: ‘()’,
right: { left: ‘”整数=”‘, op: ‘,’, right: ‘1234’ }
}
},
op: ‘;’,
right: {
left: ‘print’,
op: ‘()’,
right: { left: ‘”実数=”‘, op: ‘,’, right: ‘1234.5’ }
}
},
op: ‘;’,
right: undefined
}
引数1個目引数2個目引数3個目
引数4
整数=1234
実数=1234.5