gpt4 book ai didi

javascript - node.js 中的命名空间导入

转载 作者:搜寻专家 更新时间:2023-10-31 23:55:20 26 4
gpt4 key购买 nike

我有一些允许合并命名空间的函数,当模块包含很多函数时,非常类似于 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。

最佳答案

tl;dr:没有。

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

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

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

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

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

  1. 首先,它根据a somewhat complex set of rules 确定应该加载哪个文件。 .

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

  3. 原文为wrapped在匿名函数中。 test.js 最终会看起来像这样:

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

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

  4. 模块加载器 eval s1 第 3 步中的源文本,然后调用生成的匿名函数,传入一个新的 exports目的。 (参见 Module#_compile 。)

    1 - 真的 vm.runInThisContext ,类似于 eval除了它无权访问调用者的范围

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

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

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

例如,对于对象我们可以做如下事情:

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

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

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

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

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

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

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

  1. absolutely kills performance因为 V8 无法执行名称查找优化。
  2. 它已被弃用,并且在严格模式下被禁止。

老实说,我建议不要用 eval 做一些棘手的事情,帮您 future 的维护者一个忙,只需遵循将模块的导出分配给局部变量的标准做法即可。

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

关于javascript - node.js 中的命名空间导入,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18315029/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com