微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

JavaScript:在一个虚拟机中完成所有评估

如何解决JavaScript:在一个虚拟机中完成所有评估

我正在创建一个自定义 JavaScript 控制台,我希望它的工作方式与开发工具中的控制台完全一样。 (或者……类似 REPL 的东西)https://github.com/MohammadMD1383/js-interactive

我一一获取用户输入并对其进行评估。 eval(userInput)

问题在于定义变量。我注意到 eval 函数每次都使用一个新的 VM,所以声明是在一个单独的 VM 中而不是对变量的调用。所以它会导致错误 someVarName is not defined

我的代码示例:

button.onclick = () => {
    evaluateMyExpression(textarea.value);
};

function evaluateMyExpression(code) {
    let result = eval(code);
    // do something else …
}

enter image description here

解决方法

您可以使用生成器函数并在它们进来时向其“馈送”表达式。由于生成器保留其上下文,因此在评估下一行时,前几行中的所有变量都将存在:

function* VM(expr) {
    while (1) {
        try {
            expr = yield eval(expr || '')
        } catch(err) {
            expr = yield err
        }
    }
}


vm = VM()
vm.next()

function evaluate(line) {
    let result = vm.next(line).value
    if (result instanceof Error)
        console.log(line,'ERROR',result.message)
    else
        console.log(line,'result',result)
}

evaluate('var a = 55')
evaluate('a')
evaluate('var b = 5')
evaluate('c = a + b')
evaluate('foobar--')
evaluate('c++')
evaluate('[a,b,c]')

another thread 中所建议,“自相似”eval 也将保留块作用域变量:

let _EVAL = e => eval(`_EVAL=${_EVAL}; undefined; ${e}`)

function evaluate(line) {
    try {
      let result = _EVAL(line)
      console.log(line,'==>',result)
    } catch (err) {
        console.log(line,err.message)
    }
}

evaluate('var a = 55')
evaluate('a')
evaluate('foobar--')
evaluate('let x = 10')
evaluate('const y = 20')
evaluate('[a,x,y]')

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。