草庐IT

javascript - 什么是模块,module.exports与export之间的区别是什么?

coder 2025-02-17 原文

我已经读了几个小时的关于这个话题的书,只是没有发现任何东西可以帮助你坚持下去。

一个模块只是 Node 中具有一些属性的对象,一个是引用对象的导出属性。

“导出”变量为

var exports = module.exports

这是一个变量,指向module.exports所引用的对象。

我正在努力的是可视化模块是什么。我知道这是一个对象,但是只有一个吗?

我知道这不是 Node 实现模块的确切方式,但我正在将其可视化,如下所示:
var module = {}

module.exports = {}

// now module has a property module.exports

var exports = module.exports

现在,从我一直在阅读的所有内容中,如果您要为module.export ='xyz'分配某些内容

它将保留值“xyz”。它会丢失原始对象吗?最重要的是,如果我在同一文件中为module.exports分配了其他内容,是否将其替换为新值?
EX: 

// file = app.js

module.export = 'hello'
module.export = 'bye'

// file = newApp.js

require(./app);

该模块的值(value)是什么?我是覆盖同一个模块对象还是存在多个对象?

最佳答案

在我们继续之前,重要的是您了解how modules are actually loaded by node

从 Node 的模块加载系统中脱颖而出的关键是,在它实际运行您的require(在 Module#_compile 中发生)的代码之前,它会creates一个新的空exports对象作为Module的属性。 (换句话说,您的可视化是正确的。)

然后, Node 使用匿名函数wrapsrequire d文件中的文本:

(function (exports, require, module, __filename, __dirname) {
// here goes what's in your js file
});

...以及本质上eval的字符串。对该字符串进行eval的结果是一个函数(一个匿名包装器), Node 立即调用该函数,并传递如下参数:
evaledFn(module.exports, require, module, filename, dirname);
requirefilenamedirname是对require函数(实际上是isn't a global)以及包含已加载模块路径信息的字符串的引用。 (这就是docs的意思,当他们说“__filename实际上不是全局的,而是每个模块的局部”。)

因此,我们可以看到在模块内部,实际上是exports === module.exports。但是,为什么模块同时获得moduleexports呢?

向后兼容。在 Node 的早期,模块内部没有module变量。您只需分配给exports即可。但是,这意味着您永远无法将构造函数导出为模块本身。

一个将构造函数导出为模块的模块的示例:
var express = require('express');
var app = express();

之所以有效,是因为Express exports a function by reassigning module.exports 覆盖了 Node 默认为您提供的空对象:
module.exports = function() { ... }

但是,请注意,您不能只写:
exports = function { ... }

这是因为JavaScript's parameter passing semantics are weird。从技术上讲,JavaScript可能被认为是“纯值传递”,但实际上参数是“按值引用”传递的。

这意味着当您将对象传递给函数时,它会接收对该对象的引用作为值。您可以通过该引用访问原始对象,但不能更改调用者对引用的感知。换句话说,没有像在C/C++/C#中可能看到的“输出”参数这样的东西。

作为一个具体的例子:
var obj = { x: 1 };

function A(o) {
    o.x = 2;
}
function B(o) {
    o = { x: 2 };
}

调用A(obj);将产生obj.x == 2,因为我们访问以o传入的原始对象。但是,B(obj);不会执行任何操作。 obj.x == 1。为B的本地o分配一个全新的对象只会更改o指向的对象。它对调用者的对象obj不起作用,该对象不受影响。

现在显而易见,为什么因此有必要将module对象添加到 Node 模块的本地范围。为了允许模块完全替换exports对象,它必须作为传递给模块匿名函数的对象的属性而可用。显然没有人想要破坏现有的代码,因此保留exports作为对module.exports的引用。

因此,当您只为导出对象分配属性时,使用exports还是module.exports都没有关系;它们是一个相同的对象,指向完全相同的对象的引用。

仅当您要将函数导出为必须使用module.exports的顶级导出时,因为如我们所见,仅将函数分配给exports不会超出模块范围。

最后一点,将函数导出为模块时,最好同时分配exportsmodule.exports。这样,两个变量都保持一致,并且与标准模块中的变量相同。
exports = module.exports = function() { ... }

确保在模块文件的顶部附近执行此操作,以确保没有人无意中分配一个最终被覆盖的exports

另外,如果您觉得这很奇怪(三个= s?),那么我将利用一个事实,即包含assignment operator的表达式将返回分配的值,这样就可以一次将一个值分配给多个变量。

关于javascript - 什么是模块,module.exports与export之间的区别是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21471103/

有关javascript - 什么是模块,module.exports与export之间的区别是什么?的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby - 在 Ruby 中使用匿名模块 - 2

    假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

  3. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  4. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  5. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  6. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  7. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  8. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

    为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

  9. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  10. ruby-on-rails - 在混合/模块中覆盖模型的属性访问器 - 2

    我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah

随机推荐