3

我有一些允许合并命名空间的功能,非常类似于import模块包含大量功能时(我公开了一个带有数十个组合器的 API)它var f = target.f;为导出的每个项目生成很多

function getNamespace(name, exports){
    var output='';
    for(var item in exports){
        output += 'var ' + item + ' = '+name+ '.'+item + ';';
    }
    return output;
}

和用法:

var paco = require('./paco.js');

eval(paco.getNamespace('paco', paco));

// instead of paco.between(paco.start(),paco.content(),paco.end())
between(start(), content(), end())

问题

我有办法将评估“隐藏”到某个函数中吗?我既不想改变全局命名空间也不想调用vm.runInThisContext,只需要在调用函数之后将一些局部变量添加到调用上下文中,类似于require.

我的意思是我需要类似的东西

import('./paco'); 
// this should work like this
// var paco = require('./paco.js');       
// var between = paco.between;

但在调用范围内没有全局突变,也没有 eval。

4

3 回答 3

2

tl;博士:不。

为了理解为什么这是不可能的,了解 Node 在幕后做了什么很重要。

假设我们在 test.js 中定义了一个函数:

function foo() {
    var msg = 'Hello world';
    console.log(msg);
}

在传统的浏览器 JavaScript 中,只需将该函数声明放在一个文件中并使用<script>标签拉入该文件,就会导致foo在全局范围内声明。

require()当你创建一个文件时,Node 会做不同的事情。

  1. 首先,它根据一组有些复杂的规则准确地确定应该加载哪个文件。

  2. 假设文件是​​ JS 文本(不是已编译的 C++ 插件),Node 的模块加载器调用 fs.readFileSync以获取文件的内容。

  3. 源文本包装在一个匿名函数中。test.js 最终会看起来像这样:

    (function (exports, require, module, __filename, __dirname) {
    function foo() {
        var msg = 'Hello world';
        console.log(msg);
    }
    });
    

    对于曾经将自己的代码包装在匿名函数表达式中以防止变量泄漏到浏览器中的全局范围内的任何人来说,这应该看起来很熟悉。它还应该开始理解 Node 中的“神奇”变量是如何工作的。

  4. 模块加载器evals 1来自步骤 3 的源文本,然后调用生成的匿名函数,传入一个新exports对象。(见Module#_compile。)

    1 - 真的vm.runInThisContext,这就像eval除了它无法访问调用者的范围

  5. 匿名包装函数返回后,module.exports内部缓存 的值,然后由require. (后续调用以require()返回缓存值。)

正如我们所见,Node 通过简单地将文件的源代码包装在匿名函数中来实现“模块”。因此,不可能将函数“导入”到模块中,因为 JavaScript 不提供对函数执行上下文的直接访问——即函数局部变量的集合。

换句话说,我们无法循环访问函数的局部变量,也无法像使用对象的属性一样创建具有任意名称的局部变量。

例如,对于对象,我们可以执行以下操作:

var obj = { key: 'value' };
for (var k in obj) ...
obj[propertyNameDeterminedAtRuntime] = someValue;

但是没有代表函数的局部变量的对象,这对于我们将对象的属性(如exports模块的属性)复制到函数的局部范围中是必要的。

您所做的是在当前范围内使用eval. 生成的代码使用关键字声明局部变量var,然后将其注入到eval调用的范围内。

无法将eval调用移出模块,因为这样做会导致注入的代码插入到不同的范围内。请记住,JavaScript 具有静态作用域,因此您只能访问包含您的函数的词法作用域。

另一种解决方法是使用with,但您应该避免with使用.

with (require('./paco.js')) {
    between(start(), content(), end())
}

with不应该使用有两个原因:

  1. 绝对会影响性能,因为 V8 无法执行名称查找优化。
  2. 它已被弃用,并且在严格模式下被禁止。

老实说,我建议eval不要对 .

如果您经常键入它,请将其设为单字符名称(或使用更好的编辑器)。

于 2013-08-19T15:41:18.743 回答
0

根据这个答案Node.js 标准模块的全局变量?存在global与浏览器中相同的对象window。因此,您可以向该对象添加密钥

function getNamespace(exports) {
    for(var item in exports){
        global[item] = exports[item];
    }
}

并将其用作:

paco.getNamespace(paco);

根本不需要评估。

于 2013-08-19T13:40:37.923 回答
0

不可以。无法从外部模块修改本地范围。原因是,当在外部模块中调用 eval 时,它的上下文将是外部模块,而不是需要模块的范围。

此外,vm.runInThisContext 无权访问本地范围,因此也无济于事。

于 2013-08-19T15:18:34.783 回答