Nodejs中的require函数的具体使用方法
说明
本文参考Node官网文档版本为v11.12.0。
本文主要分析了Nodejs中require导入JSON和js文件时得到的结果,同时简单涉及到了Nodejs中模块导出module.exports和exports的用法。
引言
在阅读webpack源码的过程当中,见到如下一行代码:
constversion=require("../package.json").version
故引申出对Nodejs中require的学习。
require介绍
在Node.js的文档中,require的相关文档是在Modules目录下,属于Nodejs模块化系统的一部分。
require是一个函数。通过typeof或者Object.prototype.toString.call()可以验证这个结论:
console.log(require)//输出:Function console.log(Object.prototype.toString.call(require)//输出:[objectFunction]
通过直接打印require,可以发现在require函数下还挂载着若干个静态属性,这些静态属性也可以在Nodejs的官方文档中直接找到相关的说明:
{[Function:require] resolve:{[Function:resolve]paths:[Function:paths]}, main: Module{ id:'.', exports:{}, parent:null, filename:'/Users/bjhl/Documents/webpackSource/index.js', loaded:false, children:[], paths: ['/Users/bjhl/Documents/webpackSource/node_modules', '/Users/bjhl/Documents/node_modules', '/Users/bjhl/node_modules', '/Users/node_modules', '/node_modules']}, extensions: [Object:nullprototype]{'.js':[Function],'.json':[Function],'.node':[Function]}, cache: [Object:nullprototype]{ '/Users/bjhl/Documents/webpackSource/index.js': Module{ id:'.', exports:{}, parent:null, filename:'/Users/bjhl/Documents/webpackSource/index.js', loaded:false, children:[], paths:[Array]}}}
require函数静态属性
这里之后再详细补充。
require使用
在官网文档中可以看到如下关于require的说明:
require(id)# Addedin:v0.1.13 idmodulenameorpath Returns:exportedmodulecontent Usedtoimportmodules,JSON,andlocalfiles.Modulescanbeimportedfromnode_modules.LocalmodulesandJSONfilescanbeimportedusingarelativepath(e.g../,./foo,./bar/baz,../foo)thatwillberesolvedagainstthedirectorynamedby__dirname(ifdefined)orthecurrentworkingdirectory.
同时还给出了三种require的使用方法:
//Importingalocalmodule: constmyLocalModule=require('./path/myLocalModule'); //ImportingaJSONfile: constjsonData=require('./path/filename.json'); //Importingamodulefromnode_modulesorNode.jsbuilt-inmodule: constcrypto=require('crypto');
从以上文档中可以得出以下信息:
- require接受一个参数,形参名为id,类型是String。
- require函数return的是模块到处的内容,类型是任意。
- require函数可以导入模块、JSON文件、本地文件。模块可以通过一个相对路径从node_modules、本地模块、JSON文件中导出,该路径将针对__dirname变量(如果已定义)或者当前工作目录。
require实践
在这里将分类讨论require的实践结论。
require导入JSON
JSON是一种语法,用来序列化对象、数组、数值、字符串、布尔值和null。
在文章的开头就提到了通过require("./package.json")文件来读取package.json文件中的version属性。这里将尝试导入info.json文件并查看相关信息。
文件结构目录如下:
. ├──index.js └──info.json
将info.json文件的内容修改为:
{ "name":"myInfo", "hasFriend":true, "salary":null, "version":"v1.0.0", "author":{ "nickname":"HelloKitty", "age":20, "friends":[ { "nickname":"snowy", "age":999 } ] } }
在info.json当中,包含了字符串、布尔值、null、数字、对象和数组。
将index.js的内容修改如下并在当前terminal运行命令nodeindex.js,得到如下结果:
constinfo=require("./info.json") console.log(Object.prototype.toString.call(info))//[objectObject] console.log(info.version)//v1.0.0 console.log(info.hasFriend)//true console.log(info.salary)//null console.log(info.author.nickname)//HelloKitty console.log(info.author.friends)//[{nickname:'snowy',age:999}]
可以看到,require导入一个JSON文件的时候,返回了一个对象,Nodejs可以直接访问这个对象里的所有属性,包括String、Boolean、Number、Null、Object、Array。个人猜测这里可能用到了类似于JSON.parse()的方法。
通过这个结论也得出了一种思路,即通过require方法传入JSON文件来读取某些值,如在文章开头中,webpack通过读取package.json文件获取到了version值。
require导入本地js文件
文件结构目录如下:
. ├──index.js ├──module_a.js └──module_b.js
index.js文件中,分别按顺序导入了module_a和module_b并赋值,然后将这两个变量打印,内容如下:
console.log("***index.js开始执行***") constmodule_a=require("./module_a") constmodule_b=require("./module_b") console.log(module_a,"***打印module_a***") console.log(module_b,"***打印module_b***") console.log("***index.js结束执行***")
module_a文件中,未指定module.exports或者exports,但是添加了一个异步执行语句setTimeout,内容如下:
console.log("**module_a开始执行**") letname="I'mmodule_a" setTimeout(()=>{ console.log(name,"**setTimeout打印a的名字**") },0) console.log("**module_a结束执行**")
module_b文件中,指定了module.exports(也可以换成exports.name,但是不能直接使用exports等于某个对象,因为exports和module.exports其实是指向了一个地址,引用了相同的对象,如果使用exports等于其他的引用类型,则不再指向module.exports,无法改变module.exports里的内容),内容如下:
console.log("**module_b开始执行**") letname="I'mmodule_b" console.log(name,"**打印b的名字**") module.exports={ name } console.log("**module_b结束执行**")
在当前目录terminal下运行nodeindex.js运行得到如下输出:
***index.js开始执行***
**module_a开始执行**
**module_a结束执行**
**module_b开始执行**
Iammodule_b**打印b的名字**
**module_b结束执行**
{}'***打印module_a***'
{name:'Iammodule_b'}'***打印module_b***'
***index.js结束执行***
Iammodule_a**setTimeout打印a的名字**
通过以上执行结果可以得出结论:
- require某个js文件时,如果未通过exports或者module.exports指定导出内容,则require返回的结果是一个空对象;反之可以通过module.export或者给exports属性赋值来导出指定内容。
- require某个js文件时,该文件会立即sync执行。
require导入模块
我们先选择一个npm包——cors。进入文件夹,运行一下命令:
npminit-y//初始化 echo-e"letcors=require(\"cors\")\nconsole.log(cors)">index.js//生成index.js文件 npminstallcors--save//安装cors包
文件结构如下(...处省略了其他的模块):
. ├──index.js ├──node_modules │├──cors ││├──CONTRIBUTING.md ││├──HISTORY.md ││├──LICENSE ││├──README.md ││├──lib │││└──index.js ││└──package.json ││... ├──package-lock.json └──package.json
index.js中的内容如下:
letcors=require("cors") console.log(cors)
运行nodeindex.js,得出以下结果:
[Function:middlewareWrapper]
找到node_modules下的cors模块文件夹,观察cros模块中的package.json文件,找到main字段:"main":"./lib/index.js",找到main字段指向的文件,发现这是一个IIFE,在IIFE中的代码中添加,console.log("hellocors"),模拟代码结构如下:
(function(){ 'usestrict'; console.log("hellocors");//这是手动添加的代码 ... functionmiddlewareWrapper(o){ ... } module.exports=middlewareWrapper; })()
再次运行nodeindex.js,得出以下结果:
hellocors
[Function:middlewareWrapper]
为什么会打印出hellocors呢?因为require模块的时候,引入的是该模块package.json文件中main字段指向的文件。而这个js文件会自动执行,跟require引用本地js文件是相同的。
packjson文档
在npm的官方网站中可以找到关于package.json中的main字段定义。
main ThemainfieldisamoduleIDthatistheprimaryentrypointtoyourprogram.Thatis,ifyourpackageisnamedfoo,andauserinstallsit,andthendoesrequire("foo"),thenyourmainmodule'sexportsobjectwillbereturned. ThisshouldbeamoduleIDrelativetotherootofyourpackagefolder Formostmodules,itmakesthemostsensetohaveamainscriptandoftennotmuchelse.
在以上说明中可以得出以下结论:
- main字段是一个模块ID,是程序的主入口。
- 当使用require("xxx")的时候,导入的是main字段对应的js文件里的module.exports。
所以require导入模块的时候,是运行的对应模块package.json中main字段指定的文件。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。