在nodejs中,用户自己编写的模块,称为文件模块。
文件模块,则是指js文件、json文件或者是.node文件。在引用文件模块的时候后要加上文件的路径:/.../.../xxx.js表示绝对路径、./xxx.js表示相对路径(同一文件夹下的xxx.js),../表示上一级目录。如果既不加/.../、../又不加./的话,则该模块要么是核心模块,要么是从一个node_modules文件夹加载。
自定义模块相对路径必须写 . / ,可以省略后缀名js。
demo
在同一文件夹test下,创建a.js、b.js
分别写入内容
a.js内容:
console.log(‘a start‘)
require(‘./b‘)
console.log(‘a end‘)
b.js内容
console.log(‘b msg‘)
执行a.js,输出结果
ERROR: The process "node.exe" not found.
a start
b msg
a end
[Finished in 0.5s]
分析:require加载b.js中的代码,是执行其中的代码
文件与文件之间由于是模块作用域(在Node.js中没有全局作用域的概念),模块是完全封闭的,外部无法访问内部,内部也无法范访问外部。
在同一文件夹test下,创建a.js、b.js
分别写入内容
a.js内容:
var foo = ‘aaa‘
console.log(‘a start‘)
require(‘./b‘)
console.log(‘a end‘)
console.log(‘foo 的值是:‘, foo)
b.js内容
console.log(‘b msg‘)
var foo = ‘bbb‘
console.log(‘b end‘)
执行a.js,输出结果
ERROR: The process "node.exe" not found.
a start
b msg
b end
a end
foo 的值是: aaa
[Finished in 0.4s]
分析:从输出结果可以看出,引入的b.js并没有影响a.js中的foo的值,文件与文件之间模块作用域是隔离的
在同一文件夹test下,创建a.js、b.js
分别写入内容
a.js内容:
console.log(‘a start‘)
function add(x, y) {
return x + y
}
require(‘./b‘)
console.log(‘a end‘)
b.js内容
console.log(‘b msg‘)
console.log(add(10, 20))
console.log(‘b end‘)
执行a.js,输出结果执行报错,add is not defined,函数未被定义,无法引用原文件模块的函数
require
函数用于在当前模块中加载和使用别的模块,传入一个模块名,返回一个模块导出对象。模块名可使用相对路径(以./
开头),或者是绝对路径(以/
或C:
之类的盘符开头)。另外,模块名中的.js
扩展名可以省略。以下是一个例子。
require 方法有两个作用:
var foo1 = require(‘./foo‘);
var foo2 = require(‘./foo.js‘);
var foo3 = require(‘/home/user/foo‘);
var foo4 = require(‘/home/user/foo.js‘);
// foo1至foo4中保存的是同一个模块的导出对象。
另外,可以使用以下方式加载和使用一个JSON文件。
var data = require(‘./data.json‘);
exports
对象是当前模块的导出对象,用于导出模块公有方法和属性。别的模块通过require
函数使用当前模块时得到的就是当前模块的exports
对象。以下例子中导出了一个公有方法。
在每个文件模块中都提供了一个对象:exports
exports 默认是一个空对象
你要做的就是把所有需要被外部访问的成员挂载到这个 exports 对象中
exports.hello = function () {
console.log(‘Hello World!‘);
};
测试
在同一文件夹test下,创建a.js、b.js
分别写入内容
a.js内容:
var bExports = require(‘./b‘) //引入b文件处理模块
console.log(bExports.foo) //从require对象中接收到foo属性的值
console.log(bExports.add(10, 30))//从require对象中接收到add函数
console.log(bExports.age) //未能接收到age属性,所以输出值是undefined
bExports.readFile(‘./a.js‘) //这里执行的是b.js的readFile函数,与fs模块中的
//readFile函数无关
console.log(‘------分割线-------‘)
var fs = require(‘fs‘) //引入fs文件处理模块
fs.readFile(‘./a.js‘, function (err, data) {
//执行fs模块的readFile方法
if (err) {
console.log(‘读取文件失败‘)
} else {
console.log(data.toString())
}
})
b.js内容
var foo = ‘bbb‘
// console.log(exports)
exports.foo = ‘hello‘
exports.add = function (x, y) {
return x + y
}
exports.readFile = function (path, callback) {
console.log(‘文件路径:‘, path)
}
var age = 18
// exports.age = age
function add(x, y) {
return x - y
}
执行a.js,输出结果
ERROR: The process "node.exe" not found.
hello
40
undefined
文件路径: ./a.js
------分割线-------
var bExports = require(‘./b‘) //引入b文件处理模块
console.log(bExports.foo) //从require对象中接收到foo属性的值
console.log(bExports.add(10, 30))//从require对象中接收到add函数
console.log(bExports.age) //未能接收到age属性,所以输出值是undefined
bExports.readFile(‘./a.js‘) //这里执行的是b.js的readFile函数,与fs模块中的
//readFile函数无关
console.log(‘------分割线-------‘)
var fs = require(‘fs‘) //引入fs文件处理模块
fs.readFile(‘./a.js‘, function (err, data) {
//执行fs模块的readFile方法
if (err) {
console.log(‘读取文件失败‘)
} else {
console.log(data.toString())
}
})
[Finished in 0.4s]
分析:a.js通过bExports接收b.js通过exports`对象挂载的属性和函数,没有则显示undefined
res.setHeader()
var http = require(‘http‘)
var server = http.createServer()
server.on(‘request‘, function (req, res) {
// 在服务端默认发送的数据,其实是 utf8 编码的内容
// 但是浏览器不知道你是 utf8 编码的内容
// 浏览器在不知道服务器响应内容的编码的情况下会按照当前操作系统的默认编码去解析
// 中文操作系统默认是 gbk
// 解决方法就是正确的告诉浏览器我给你发送的内容是什么编码的
// 在 http 协议中,Content-Type 就是用来告知对方我给你发送的数据内容是什么类型
// res.setHeader(‘Content-Type‘, ‘text/plain; charset=utf-8‘)
// res.end(‘hello 世界‘)
var url = req.url
if (url === ‘/plain‘) {
// text/plain 就是普通文本
res.setHeader(‘Content-Type‘, ‘text/plain; charset=utf-8‘)
//设置响应头
res.end(‘hello 世界‘)
} else if (url === ‘/html‘) {
// 如果你发送的是 html 格式的字符串,则也要告诉浏览器我给你发送是 text/html 格式的内容
res.setHeader(‘Content-Type‘, ‘text/html; charset=utf-8‘)
res.end(‘<p>hello html <a href="">点我</a></p>‘)
}
})
server.listen(3000, function () {
console.log(‘Server is running...‘)
})
通过网络发送文件( http结合fs发送文件中的数据):发送的并不是文件,本质上发送的是文件的内容,当浏览器收到服务器响应内容之后,就会根据Content-Type进行对应的解析处理
// 1. 结合 fs 发送文件中的数据
// 2. Content-Type
// http://tool.oschina.net/commons
// 不同的资源对应的 Content-Type 是不一样的
// 图片不需要指定编码
// 一般只为字符数据才指定编码
var http = require(‘http‘)
var fs = require(‘fs‘)
var server = http.createServer()
server.on(‘request‘, function (req, res) {
// / index.html
var url = req.url
if (url === ‘/‘) {
// 肯定不这么干
// res.end(‘<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><title>Document</title></head><body><h1>首页</h1></body>/html>‘)
// 我们要发送的还是在文件中的内容
fs.readFile(‘./resource/index.html‘, function (err, data) {
if (err) {
res.setHeader(‘Content-Type‘, ‘text/plain; charset=utf-8‘)
res.end(‘文件读取失败,请稍后重试!‘)
} else {
// data 默认是二进制数据,可以通过 .toString 转为咱们能识别的字符串
// res.end() 支持两种数据类型,一种是二进制,一种是字符串
res.setHeader(‘Content-Type‘, ‘text/html; charset=utf-8‘)
res.end(data)
}
})
} else if (url === ‘/xiaoming‘) {
// url:统一资源定位符
// 一个 url 最终其实是要对应到一个资源的
fs.readFile(‘./resource/ab2.jpg‘, function (err, data) {
if (err) {
res.setHeader(‘Content-Type‘, ‘text/plain; charset=utf-8‘)
res.end(‘文件读取失败,请稍后重试!‘)
} else {
// data 默认是二进制数据,可以通过 .toString 转为咱们能识别的字符串
// res.end() 支持两种数据类型,一种是二进制,一种是字符串
// 图片就不需要指定编码了,因为我们常说的编码一般指的是:字符编码
res.setHeader(‘Content-Type‘, ‘image/jpeg‘)
res.end(data)
}
})
}
})
server.listen(3000, function () {
console.log(‘Server is running...‘)
})
使用两个空格?– 进行缩进
字符串使用单引号?– 需要转义的地方除外
不再有冗余的变量?– 这是导致?大量?bug 的源头!
行首不要以?(,?[, or?`?开头
这是省略分号时唯一会造成问题的地方 –?工具里已加了自动检测!
关键字后加空格?if (condition) { ... }
函数名后加空格?function name (arg) { ... }
坚持使用全等?===?摒弃?==?一但在需要检查?null || undefined?时可以使用?obj == null。
一定要处理 Node.js 中错误回调传递进来的?err?参数。
使用浏览器全局变量时加上?window?前缀 –?document?和?navigator?除外
避免无意中使用到了这些命名看上去很普通的全局变量,?open,?length, event?还有?name。
// 当你采用了无分号的代码风格的时候,只需要注意以下情况就不会有上面的问题了:
// 当一行代码是以:
// (
// [
// `
// 开头的时候,则在前面补上一个分号用以避免一些语法解析错误。
// 所以你会发现在一些第三方的代码中能看到一上来就以一个 ; 开头。
// 结论:
// 无论你的代码是否有分号,都建议如果一行代码是以 (、[、` 开头的,则最好都在其前面补上一个分号。
// 有些人也喜欢玩儿一些花哨的东西,例如可以使用 ! ~ 等。
function hello (name) {
console.log(‘hi‘, name)
}
console.log(‘hello there‘)
$("<div class=‘box‘>")
function myFunction () {
var result = something() // ? avoid
}
if (condition) { ... } // ? ok
if(condition) { ... } // ? avoid
function name (arg) { ... } // ? ok
function name(arg) { ... } // ? avoid
run(function () { ... }) // ? ok
run(function() { ... }) // ? avoid
例外:obj == null
可以用来检查 null || undefined
if (name === ‘John‘) // ? ok
if (name == ‘John‘) // ? avoid
if (name !== ‘John‘) // ? ok
if (name != ‘John‘) // ? avoid
// ? ok
var x = 2
var message = ‘hello, ‘ + name + ‘!‘
// ? avoid
var x=2
var message = ‘hello, ‘+name+‘!‘
// ? ok
var list = [1, 2, 3, 4]
function greet (name, options) { ... }
// ? avoid
var list = [1,2,3,4]
function greet (name,options) { ... }
// ? ok
if (condition) {
// ...
} else {
// ...
}
// ? avoid
if (condition) {
// ...
}
else {
// ...
}
// ? ok
if (options.quiet !== true) console.log(‘done‘)
// ? ok
if (options.quiet !== true) {
console.log(‘done‘)
}
// ? avoid
if (options.quiet !== true)
console.log(‘done‘)
例外: document
, console
, navigator
window.alert(‘hi‘) // ? ok
// ? ok
var value = ‘hello world‘
console.log(value)
// ? avoid
var value = ‘hello world‘
console.log(value)
// ? ok
var location = env.development ? ‘localhost‘ : ‘www.api.com‘
// ? ok
var location = env.development
? ‘localhost‘
: ‘www.api.com‘
// ? avoid
var location = env.development ?
‘localhost‘ :
‘www.api.com‘
// ? ok
var silent = true
var verbose = true
// ? avoid
var silent = true, verbose = true
// ? avoid
var silent = true,
verbose = true
function foo () {return true} // ? avoid
function foo () { return true } // ? ok
function my_function () { } // ? avoid
function myFunction () { } // ? ok
var my_var = ‘hello‘ // ? avoid
var myVar = ‘hello‘ // ? ok
var obj = {
message: ‘hello‘, // ? avoid
}
var obj = {
foo: ‘foo‘
,bar: ‘bar‘ // ? avoid
}
var obj = {
foo: ‘foo‘,
bar: ‘bar‘ // ? ok
}
console.
log(‘hello‘) // ? avoid
console
.log(‘hello‘) // ? ok
文件末尾留一空行
函数调用时标识符与括号间不留间隔
console.log (‘hello‘) // ? avoid
console.log(‘hello‘) // ? ok
var obj = { ‘key‘ : ‘value‘ } // ? avoid
var obj = { ‘key‘ :‘value‘ } // ? avoid
var obj = { ‘key‘:‘value‘ } // ? avoid
var obj = { ‘key‘: ‘value‘ } // ? ok
function animal () {}
var dog = new animal() // ? avoid
function Animal () {}
var dog = new Animal() // ? ok
var nums = new Array(1, 2, 3) // ? avoid
var nums = [1, 2, 3] // ? ok
const score = 100
score = 125 // ? avoid
if (false) { // ? avoid
// ...
}
if (x === 0) { // ? ok
// ...
}
while (true) { // ? ok
// ...
}
import { myFunc1 } from ‘module‘
import { myFunc2 } from ‘module‘ // ? avoid
import { myFunc1, myFunc2 } from ‘module‘ // ? ok
Object.prototype.age = 21 // ? avoid
switch (filter) {
case 1:
doSomething() // ? avoid
case 2:
doSomethingElse()
}
switch (filter) {
case 1:
doSomething()
break // ? ok
case 2:
doSomethingElse()
}
switch (filter) {
case 1:
doSomething()
// fallthrough // ? ok
case 2:
doSomethingElse()
}
window = {} // ? avoid
不要混合使用空格与制表符作为缩进
除了缩进,不要使用多个空格
const id = 1234 // ? avoid
const id = 1234 // ? ok
new Character() // ? avoid
const character = new Character() // ? ok
var sum = new Function(‘a‘, ‘b‘, ‘return a + b‘) // ? avoid
let config = new Object() // ? avoid
const pathToFile = __dirname + ‘/app.js‘ // ? avoid
const pathToFile = path.join(__dirname, ‘app.js‘) // ? ok
let name = ‘John‘
let name = ‘Jane‘ // ? avoid
let name = ‘John‘
name = ‘Jane‘ // ? ok
function sum (a, b) {
return result = a + b // ? avoid
}
function sum (a, b) {
return (result = a + b) // ? ok
}
function doSomething () {
return true
console.log(‘never called‘) // ? avoid
}
fn(... args) // ? avoid
fn(...args) // ? ok
for (let i = 0 ;i < items.length ;i++) {...} // ? avoid
for (let i = 0; i < items.length; i++) {...} // ? ok
if (admin){...} // ? avoid
if (admin) {...} // ? ok
getName( name ) // ? avoid
getName(name) // ? ok
//comment // ? avoid
// comment // ? ok
/*comment*/ // ? avoid
/* comment */ // ? ok
const message = `Hello, ${ name }` // ? avoid
const message = `Hello, ${name}` // ? ok
if (price === NaN) { } // ? avoid
if (isNaN(price)) { } // ? ok
window.alert(‘hi‘) // ? ok
window.alert(‘hi‘); // ? avoid
(
, [
, or ``` 等作为一行的开始。在没有分号的情况下代码压缩后会导致报错,而坚持这一规范则可避免出错。// ? ok
;(function () {
window.alert(‘ok‘)
}())
// ? avoid
(function () {
window.alert(‘ok‘)
}())
// ? ok
;[1, 2, 3].forEach(bar)
// ? avoid
[1, 2, 3].forEach(bar)
// ? ok
;`hello`.indexOf(‘o‘)
// ? avoid
`hello`.indexOf(‘o‘)
上面的写法只能说聪明过头了
比如:
;[1, 2, 3].forEach(bar)
建议的写法是:
var nums = [1, 2, 3]
nums.forEach(bar)
参考资料
[1]https://www.jianshu.com/p/e889069b7c27
[2]https://github.com/standard/standard/blob/master/docs/RULES-zhcn.md
nodejs(第三篇):nodejs中的文件模块、nodejs中的require与exports、http模块补充、JavaScript Standard Style
原文:https://www.cnblogs.com/Nicholas0707/p/12571457.html