(一)模块的分类
我们知道,在 node.js 中,通过 require 来引入外部模块,引入外部模块需要使用模块标识,通过模块标识来找到指定的模块。外部模块一般分为两大类:核心模块和文件模块
(1)核心模块
由Node本身提供的模块,模块标识是模块的名字
var path = require('path')
(2)文件模块
由用户自己创建的模块,模块的标识是模块的路径
const ENV = require('../src/utils/env')
(二)模块的本质
如何验证模块是函数呢?函数有一个标识,就是函数具有arguments,如果能打印出arguments,则说明是函数
模块的本质是一个函数,一个模块的变量如果不暴露出去的话,其他模块是无法访问的,下面就对模块进行详细解析
如何获取函数本身?当然是使用arguments.callee啦~我们在 js 中输入下面代码,然后 node index.js,看看能获得什么?
console.log(arguments.callee) // [Function]
结果输出一个 [Function],说明一个模块就是一个函数,那么函数是什么呢?
我们直达require是一个方法,他接受模块标识作为参数,返回模块内容,我们来检测一下
var a = 10;
console.log(arguments.callee + '')
然后 node index.js,输出
function (exports, require, module, __filename, __dirname) {
var a = 10;
console.log(arguments.callee + '')
}
破案了,node 模块会给我们写的代码中自动包装一个函数,下面我们就来一一看看每个参数都是什么意思
(1)require
一个函数,用来引入外部的模块
(2)exports
改对象用来将变量或者函数暴露到外部
(3)module
代表的就是模块本身
console.log(module.exports === exports) // true
也就是说可以使用exports导出,也可以使用module.exports 导出
(4)__filename
当前模块的路径
console.log(__filename) // /Users/macbook/Desktop/complier/index.js
(5)__dirname
当前模块所在文件夹的完整路径(父级的路径)
console.log(__dirname) // /Users/macbook/Desktop/complier
(三)__dirname 和 __filename
在 node 文件模块中,使用相对路径是不可靠的,因为 相对路径被认定为 相对于 Node 执行命令所处的路径
下面来看一下:
index.js:
const fs = require('fs')
fs.readFile('./test.txt', (err, data) => {
if (err) {
console.log(err)
return
}
console.log(data.toString())
})
输出:hello TextDecoder
如果我改变 Node 的命令所处路径,退后一层,就发现找不到了
在 node 文件模块中, 相对路径被认定为 相对于 Node 执行命令所处的路径,当Node 命令所处的路径是 complier/file时,index.js 中的 ./,./ 是指当前目录,也就是 complier/file 下的test.txt,可以运行成功,那么如果我改变Node所处命令行路径,在 compiler 中执行时,./ 就成为了 complier/test.txt,所以报错找不到。我们应该使用绝对路径,这样更安全。改造下:
const fs = require('fs')
const path = require('path')
fs.readFile(path.join(__dirname, 'test.txt'), (err, data) => {
if (err) {
console.log(err)
return
}
console.log(data.toString())
})
path.join():路径拼接
__dirname: 文件所在的目录
但需要注意的是,require 里的相对路径是相对于所在的文件
(四)模块化
总结:
(1)在 node 中,一个js 文件就是一个模块
(2)在 node 中,每一个 js 文件中的代码都是独立运行在一个函数中
(四)其他
传统的服务起都是多线程的,进来一个请求,分配一个线程给它,但是 node 是单线程的,但是在后台有一个 I/O处理池