欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Node.js模块导入导出

程序员文章站 2022-10-04 20:52:24
这篇文章本来是想模块导入导出和事件循环一起写的,但是感觉一起写的话会太长了,所以就分开两篇文章写吧。下一篇会重点介绍一下js中的事件循环,js代码到底是以何种顺序去执行的呢?我相信你看懂了事件循环再去看node对你的帮助是非常大的。 讲模块系统之前先认识一下node.js中的全局对象。 node.j ......

这篇文章本来是想模块导入导出和事件循环一起写的,但是感觉一起写的话会太长了,所以就分开两篇文章写吧。下一篇会重点介绍一下js中的事件循环,js代码到底是以何种顺序去执行的呢?我相信你看懂了事件循环再去看node对你的帮助是非常大的。

讲模块系统之前先认识一下node.js中的全局对象。

node.js的全局对象

  众所周知的是在浏览器中的老大哥是谁,它就是window,this指向的也是window,那么在node中的全局对象就不是window了,而是global,可以在命令行中去看一下,想学习node的应该已经安装了node环境,如果还没有安装可以去去找到你对应的操作系统和版本去下载,如果node命令不是全局还需要配置一下环境变量,现在window操作系统安装上node之后应该就自动配置完成了。

  打开命令行,输入 node 回车,然后输入 this 或者global就可以看到全局对象。你会看到好多东西,但是他比window对象是少太多太多了。

  在javascript中,使用script标签去引入js文件的话,那么在js文件中的全局变量都会挂载到window对象下面,在各个文件中都可以共享它那个变量,比如jquery,你引入了一个jquery文件,那么在其它的文件当中,你是可以访问到$这个变量的。

  而在node.js中是如何实现文件之间的的引入呢,就不得不提及到commonjs了。

common.js

  查阅资料是这么说的:javascript是一种强大的面向对象语言,它有很多快速高效的解释器。官方javascript标准定义的api是为了构建基于浏览器的应用程序。然而,并没有定义一个用于更广泛的应用程序的标准库。commonjs api定义了很多普通应用程序(主要指非浏览器的应用)使用的api,从而填补了这个空白。它的终极目标是提供一个类似python,ruby和java的标准库。这样的话,开发者可以使用commonjs api编写应用程序,然后这些应用可以运行在不同的javascript解释器和不同的主机环境中。在兼容 commonjs 的系统中,你可以使用javascript开发服务器端javascript应用程序、命令行工具、图形界面应用程序和hybrid混合应用程序。
 
  通俗易懂的来说就是 commonjs就是为了js的表现来指定来指定规范,因为js没有模块功能所以commonjs应运而生,它的出现,目的就是为了让js在其他环境也能执行,而不仅仅局限于浏览器。

  commonjs是一种规范,node.js就是这种规范的实现。

引入模块require

  全局变量在所有模块中均可使用。讲道理我们理解的require就是一个全局变量, 但是官方文档说的是此变量虽然看起来像全局变量,但实际上不是。 它们的作用域只在模块内。那我也不管它了,它的实现大概也就是像那种在文件中的一个函数把require传进来,像下面这样。

  具体的讲解:commonjs模块规范,我们知道每个模块文件中存在着require、exports、module这3个变量,但是它们在模块文件中并没有定义,那么从何而来呢?甚至在node的api文档中我们知道每个模块中还有__filename、__dirname这两个变量的存在,它们又是从何而来的呢?

  事实上,在编译的过程中,node.js对获取的javascript文件内容进行了头尾包装。在头部添加了(function(exports,require,module,__filename,__dirname){\n,在尾部添加了\n});一个正常的javascript文件被包装成了如下的样子。

(function (exports,require,module,__filename,__dirname) {
    exports.a = 1;
    exports.fn = function () {
        console.log(1);
    };
});

 如图为global的全局变量

Node.js模块导入导出

 

  具体的使用方法

let obj = require('./2.js');
console.log(obj);

  值得注意的是:  1.   ./代表的是当前目录下,然后是2.js,需要注意的是,如果引入的是本地的文件,那么一定要带上路径。

  2. 如果后缀名是js文件的话是可以省略的。

  3.有一些模块是不需要带路径的,它们称之为核心模块,何为核心模块

    第一种是安装好node就有的一些模块,另外一种是用npm安装依赖的那些在node_modules文件夹下面的

  4.模块的加载机制: 文件名 > 文件名.js >文件名.json>文件名.node

  与前端浏览器会缓存静态脚本文件以提高性能一样,node对引入过的模块都会进行缓存,以减少二次引入时的开销。不同的地方在于,浏览器仅仅缓存,而node缓存的是编译和执行之后的对象。

  安装好node就有的一些模块可以去node中文网的文档左侧都是,比如这些都是。

  Node.js模块导入导出

导出模块 module.exports

  先看下面的例子,运行app.js,具体操作为命令行打开到当前的目录下, 运行  node app.js

app.js

let obj = require('./2.js');
console.log(obj);   // 1

2.js

module.exports = 1;

  app.js文件中引入的是2.js文件,然后 2.js文件通过module.exports来赋值为1,require的话还有一点就是他会去寻找引入进来文件的module.export,然后把那个1赋值给了obj,打印出来 1。

  module也是一个全局变量,它和require一样,但实际上不是。 它们的作用域只在模块内。导出模块可能有的人见过下面的写法。

2.js

exports.a = 1;

 app.js代码同样不改变,打印出来的东西是  {a: 1};

其实exports是module.exports的一个引用, exports 也是一个全局变量,和上面require,module一样,作用域只在模块内。

导出模块的话你可能会想到这样做

1 module.exports = {}; //在内部的话这个东西默认等于{}
2 
3  //所以你想到了导出的话可以这样
4 module.exports = {
5   a: 1,
6   b: function(){},
7   c: 'name'
8 }

如果你这样做的话就是不正确的

 1 //这样做是不正确的
 2 exports = {
 3     a: 1,
 4     b: function(){},
 5     c: 'name'
 6 };
 7 
 8 // 正确的做法应该是这样
 9 exports.a = 1;
10 exports.b = function () {};
11 exports.c = 'name';
12 
13 // 因为exports是module.exports的一个引用,重新给exports赋值的话只会改变exports的值,而require引入的话寻找的是模块中的module.exports

  可能有的人会这样想,我直接写到global对象下面去不就可以引入了吗,为什么还需要用exports这种东西呢?具体写法如下

app.js

let obj = require('./2.js');
console.log(global.obj);
console.log(global.obj2);

2.js

global.obj = {
    a: 'name',
    b: function () {
        console.log('1111')
    }
};
global.obj2 = '狗蛋';

  怎么说呢,这样做是可以的,但是不推荐把所有的东西都写到global下面,会污染global对象,还是推荐使用exports去进行模块之间的导入导出。

   

  模块的导入导出的操作非常简答,但是对一些概念性的东西理解之后我相信会更好,当然require还有好多原理没有讲到,如果有兴趣的可以查阅资料继续学习,我相信如果只是使用的话掌握到现在的程度已经是可以的了,想继续深入的可以看看  朴灵写的 深入浅出node.js 这本书。

  如果你看了我的文章有了一些收获我会非常高兴的,由于能力有限,文章有的部分解释的不到位,希望在以后的日子里能慢慢提高自己能力,如果不足之处,还望指正。