我们正在构建一个小型 REPL,它会在用户输入时评估(使用 eval
)javascript 表达式。由于整个事情是事件驱动的,因此评估必须在单独的函数中进行,但必须在调用之间保留上下文(即所有声明的变量和函数)。我想出了以下解决方案:
function* _EVAL(s) {
while (1) {
try {
s = yield eval(s)
} catch(err) {
s = yield err
}
}
}
let _eval = _EVAL()
_eval.next()
function evaluate(expr) {
let result = _eval.next(expr).value
if (result instanceof Error)
console.log(expr, 'ERROR:', result.message)
else
console.log(expr, '===>', result)
}
evaluate('var ten = 10')
evaluate('function cube(x) { return x ** 3 }')
evaluate('ten + cube(3)')
evaluate('console.log("SIDE EFFECT")')
evaluate('let twenty = 20')
evaluate('twenty + 40') // PROBLEM
如您所见,它适用于函数范围的变量(
var
和 function
),但在 block 范围的变量( let
)上失败。我怎样才能写一个保留上下文的
eval
也可以保留 block 范围变量的包装器? 代码在浏览器中运行,DOM 和 Workers 完全可用。
应该提到的是,所需的函数必须正确处理副作用,即每一行代码,或者至少每个副作用,应该只执行一次。
链接:
JavaScript: do all evaluations in one vm | https://vane.life/2016/04/03/eval-locally-with-persistent-context/
最佳答案
article you linked包含一个实际有效的疯狂方法:在每个 eval()
期间调用,我们在 eval
内部创建一个新的闭包范围并将其导出,以便我们可以使用它来评估下一条语句。
var __EVAL = s => eval(`void (__EVAL = ${__EVAL.toString()}); ${s}`);
function evaluate(expr) {
try {
const result = __EVAL(expr);
console.log(expr, '===>', result)
} catch(err) {
console.log(expr, 'ERROR:', err.message)
}
}
evaluate('var ten = 10')
evaluate('function cube(x) { return x ** 3 }')
evaluate('ten + cube(3)')
evaluate('console.log("SIDE EFFECT")')
evaluate('let twenty = 20')
evaluate('twenty + 40') // NO PROBLEM :D
关于javascript - 上下文保留评估,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67322922/