概要
javascript虽然一直都能做服务器端程式设计词汇,但它更多的是以应用流程程式设计词汇来展现在观者眼前的。或许javascript自己都忘掉了还能做服务器端程式设计,直至2009年nodejs的陨落。
nodejs的发展史
javascript作为两门如前所述词汇,是不须要像C或是C++那般进行校对的。但在晚期的这时候,javascript发动机的继续执行工作效率是较为低的,因此引致javascript根本无法罢了dom操作方式。
随着ajax的蓬勃发展和当代web2.0的技术的产业发展,非主流应用流程地产商尽量的提高javascript的继续执行工作效率,最后Chrome V8出现了,Chrome V8是 Chromium 工程项目开放源码的 JavaScript 发动机,使javascript的继续执行工作效率得到了很大的提高。
nodejs话虽如此V8脱胎换骨了。
的。而且两门词汇能通用型其间端是何等的有诱惑力。
nodejs从2009年产业发展到2020年的nodejs 14,历经了11年的发展史,和它的父辈javascript较之还是很年青,但即使其发展性和发展性,nodejs在以两个十分快的速率往前产业发展。
nodejs概要
nodejs借助V8发动机和几组触发器的 I/O 原生植物机能,很大的提高了nodejs的处置工作效率。
触发器IO他们我们应该都很确切,和并行IO较之,缓存不必堵塞,能去处置其他枭女的事。只是在积极响应回到的这时候恢复正常操作方式,因此不会浪费CPU时间。
他们单纯看呵呵nodejs的IO数学模型:
两个好的词汇须要有较好的自然生态系品驭型,即使词汇这类根本无法提供最基本的一些操作方式,他们还须要服务器端控制系统来多样这个词汇的自然生态。
而nodejs的npm库房,代销着亚洲地区最大的开放源码库自然生态系。
基本上上使用nodejs你能实现绝大多数须要的机能。
nodejs的另外两个特点就是简单,考虑呵呵他们最常用的web应用,如果用java来写,十分麻烦,你还须要两个web服务器。
在nodejs中,一切都是那么的单纯:
const http = require(http)
const hostname = 127.0.0.1
const port = 3000
const server = http.createServer((req, res) => {
res.statusCode = 200
res.setHeader(Content-Type, text/plain)
res.end(welcome to www.flydean.com\n)
})
server.listen(port, hostname, () => {
console.log(`please visit http://${hostname}:${port}/`)
})
上面的代码就创建了两个web服务,监听在3000端口,
他们首先引入了http模块,用来进行http处置。
接着使用http 的 createServer() 方法会创建新的 HTTP 服务器并回到它。
在createServer方法内部,他们能设定要回到的对象。
最后启用server.listen机能,来监听特定的端口和服务器,当服务就绪之后,会调用后面的回调函数,继续执行特定的命令。
每当接收到新的请求的这时候,就会触发request事件,request事件能传递两个参数:
request 是两个http.IncomingMessage对象,提供了请求的详细信息。response 是两个http.ServerResponse对象,用于回到数据给调用方。在上面的例子中,他们并没有使用request,而是使用response直接构建了回到的对象。
他们设置了statusCode和header,最后使用end来关闭积极响应。
这就是两个单纯使用的nodejs流程。
nodejs的运行环境
nodejs作为js的一种,是一种如前所述词汇,一般如前所述词汇都有两种运行方式。
一种是直接运行,一种是开启两个如前所述的环境,在其中运行,nodejs也不例外。
直接运行很单纯,他们写好nodejs的流程之后,比如app.js,直接这样运行:
node app.js
如果直接继续执行node命令,就会开启REPL模式:
node
Welcome to Node.js v12.13.1.
Type “.help” for more information.
>
REPL 也被称为运行评估打印循环,是一种程式设计词汇环境(主要是控制台窗口),它使用单个表达式作为用户输入,并在继续执行后将结果回到到控制台。
REPL有什么作用呢?
第一,他们能直接在REPL中运行某些测试方法,已验证输出结果。
比如这样:
> console.log(www.flydean.com);
www.flydean.com
除此之外REPL还有一些更加有用的机能,他们知道JS中一切皆对象,比如上面他们提到的http对象,如果他们想知道http对象的大概结构怎么办呢?
直接在REPL环境中输入http即可:
> http
{
_connectionListener: [Function: connectionListener],
METHODS: [
ACL, BIND, CHECKOUT,
CONNECT, COPY, DELETE,
GET, HEAD, LINK,
LOCK, M-SEARCH, MERGE,
MKACTIVITY, MKCALENDAR, MKCOL,
MOVE, NOTIFY, OPTIONS,
PATCH, POST, PROPFIND,
PROPPATCH, PURGE, PUT,
REBIND, REPORT, SEARCH,
SOURCE, SUBSCRIBE, TRACE,
UNBIND, UNLINK, UNLOCK,
UNSUBSCRIBE
],
STATUS_CODES: {
100: Continue,
101: Switching Protocols,
102: Processing,
103: Early Hints,
200: OK,
201: Created,
202: Accepted,
203: Non-Authoritative Information,
204: No Content,
205: Reset Content,
206: Partial Content,
207: Multi-Status,
208: Already Reported,
226: IM Used,
300: Multiple Choices,
301: Moved Permanently,
302: Found,
303: See Other,
304: Not Modified,
305: Use Proxy,
307: Temporary Redirect,
308: Permanent Redirect,
400: Bad Request,
401: Unauthorized,
402: Payment Required,
403: Forbidden,
404: Not Found,
405: Method Not Allowed,
406: Not Acceptable,
407: Proxy Authentication Required,
408: Request Timeout,
409: Conflict,
410: Gone,
411: Length Required,
412: Precondition Failed,
413: Payload Too Large,
414: URI Too Long,
415: Unsupported Media Type,
416: Range Not Satisfiable,
417: Expectation Failed,
418: “Im a Teapot”,
421: Misdirected Request,
422: Unprocessable Entity,
423: Locked,
424: Failed Dependency,
425: Unordered Collection,
426: Upgrade Required,
428: Precondition Required,
429: Too Many Requests,
431: Request Header Fields Too Large,
451: Unavailable For Legal Reasons,
500: Internal Server Error,
501: Not Implemented,
502: Bad Gateway,
503: Service Unavailable,
504: Gateway Timeout,
505: HTTP Version Not Supported,
506: Variant Also Negotiates,
507: Insufficient Storage,
508: Loop Detected,
509: Bandwidth Limit Exceeded,
510: Not Extended,
511: Network Authentication Required
},
Agent: [Function: Agent] { defaultMaxSockets: Infinity },
ClientRequest: [Function: ClientRequest],
IncomingMessage: [Function: IncomingMessage],
OutgoingMessage: [Function: OutgoingMessage],
Server: [Function: Server],
ServerResponse: [Function: ServerResponse],
createServer: [Function: createServer],
get: [Function: get],
request: [Function: request],
maxHeaderSize: [Getter],
globalAgent: [Getter/Setter]
}
直接输出了http对象的简洁结构,他们还能使用tab按钮来自动补全http的方法:
> http.
http.__defineGetter__ http.__defineSetter__ http.__lookupGetter__ http.__lookupSetter__ http.__proto__ http.constructor
http.hasOwnProperty http.isPrototypeOf http.propertyIsEnumerable http.toLocaleString http.toString http.valueOf
http.Agent http.ClientRequest http.IncomingMessage http.METHODS http.OutgoingMessage http.STATUS_CODES
http.Server http.ServerResponse http._connectionListener http.createServer http.get http.globalAgent
http.maxHeaderSize http.request
PREL还支持一些特定的点操作方式:
> .help
.break Sometimes you get stuck, this gets you out
.clear Alias for .break
.editor Enter editor mode
.exit Exit the repl
.help Print this help message
.load Load JS from a file into the REPL session
.save Save all evaluated commands in this REPL session to a file
PERL还有两个特殊变量 _ ,如果在某些代码之后输入 _,则会打印最后一次操作方式的结果。process
process 对象是两个全局变量,提供了有关当前 Node.js 进程的信息并对其进行控制。 作为全局变量,它始终可供 Node.js 应用流程使用,无需使用 require()。 它也能使用 require() 显式地访问。
即使process代表的是nodejs的进程信息,因此能处置进程终止,读取环境变量,接收命令行参数等作用。
终止进程
先看呵呵怎么使用process来终止进程:
process.exit(0)
0表示正常退出,当然,他们能传入不同的退出码,表示不同的含义。
正常情况下,如果没有触发器操作方式正在等待,那么 Node.js 会以状态码 0 退出,其他情况下,会用如下的状态码:
1 未捕获异常 – 两个未被捕获的异常, 并且没被 domain 或 ‘uncaughtException’ 事件处置器处置。
2 – 未被使用 (Bash 为防内部滥用而保留)
3 内部的 JavaScript 解析错误 – Node.js 内部的 JavaScript 源代码在引导进程中引致了两个语法解析错误。一般只会在合作开发 Node.js 这类的这时候出现。
4 内部的 JavaScript 继续执行失败 – 引导进程继续执行 Node.js 内部的 JavaScript 源代码时,回到函数值失败。一般只会在合作开发 Node.js 这类的这时候出现。
5 致命错误 – 在 V8 中有两个致命的错误。 较为典型的是以 FATALERROR 为前缀从 stderr 打印出来的消息。
6 非函数的内部异常处置 – 发生了两个内部异常,但内部异常处置函数被设置成了两个非函数,或是不能被调用。
7 内部异常处置运行时失败 – 有两个不能被捕获的异常,在试图处置这个异常时,处置函数这类抛出了两个错误。比如, 如果两个 ‘uncaughtException’ 或是 domain.on(‘error’) 处置函数抛出了两个错误。
8 – 未被使用,在之前版本的 Node.js, 退出码 8 有这时候表示两个未被捕获的异常。
9 – 不可用参数 – 某个未知选项没有确定,或是没给必须要的选项填值。
10 内部的 JavaScript 运行时失败 – 调用引导函数时,引导进程继续执行 Node.js 内部的 JavaScript 源代码抛出错误。 一般只会在合作开发 Node.js 这类的这时候出现。
12 不可用的调试参数
13 未完成的Top-Level Await: await传入的Promise一直没有调用resolve方法
128 退出信号 – 如果 Node.js 接收到致命信号, 诸如 SIGKILL 或 SIGHUP,那么它的退出代码将是 128 加上信号的码值。 例如,信号 SIGABRT 的值为 6,因此预期的退出代码将为 128 + 6 或 134。
他们能通过process的on方法,来监听信号事件:
process.on(SIGTERM, () => {
server.close(() => {
console.log(进程已终止)
})
})
什么是信号?信号是两个 POSIX 内部通信控制系统:发送通知给进程,以告知其发生的事件。或是他们能从流程内部发送这个信号:
process.kill(process.pid, SIGTERM)
env
即使process进程是和外部环境打交道的,process提供了env属性,该属性承载了在启动进程时设置的所有环境变量。
默认情况下,env中的NODE_ENV被设置为development。
process.env.NODE_ENV // “development”
他们能通过修改这个环境变量,来切换nodejs的不同运行环境。
argv
process提供了argv来接收外部参数。
比如:
node app.js joe
argv是两个包含所有命令行调用参数的数组。
上面的例子中,第两个参数是 node 命令的完整路径。第二个参数是正被继续执行的文件的完整路径。所有其他的参数从第三个位置开始。
他们能这样做:
const args = process.argv.slice(2)
args[0]
如果是key=value的情况,他们能这样传参数,并且使用minimist 库来处理参数:
node app.js –name=joe
const args = require(minimist)(process.argv.slice(2))
args[name] //joe
CLI交互
const readline = require(readline).createInterface({
input: process.stdin,
output: process.stdout
})
readline.question(`how are you?`, answer => {
console.log(`${answer}!`)
readline.close()
})
如果须要更加复杂的操作方式,则能使用Inquirer.js:
const inquirer = require(inquirer)
var questions = [
{
type: input,
name: hello,
message: “how are you?”
}
]
inquirer.prompt(questions).then(answers => {
console.log(`${answers[hello]}!`)
})
exports模块
nodejs拥有内置的模块控制系统,当他们须要使用其他lib提供的机能这时候,他们能使用require来引入其他lib公开的模块。
但前提是该lib须要公开,也就是exports对应的模块出来。
nodejs的对象导出有两种方式module.exports和将对象添加为 exports 的属性。
先看第一种方式,square 模块定义在 square.js 中:
module.exports = class Square {
constructor(width) {
this.width = width;
}
area() {
return this.width ** 2;
}
};
下面的例子中, bar.js 使用了导出 Square 类的 square 模块:
const Square = require(./square.js);
const mySquare = new Square(2);
console.log(`mySquare 的面积是 ${mySquare.area()}`);
再看第二种方式,定义两个circle.js:
const { PI } = Math;
exports.area = (r) => PI * r ** 2;
exports.circumference = (r) => 2 * PI * r;
使用:
const circle = require(./circle.js);
console.log(`半径为 4 的圆的面积是 ${circle.area(4)}`);
两者都能导出特定的模块,但module.exports只会导出特定的对象,而exports是将对象添加为exports的属性,他们还须要根据属性名称来查找对象的属性。
nodejs API
除了他们上面提到的http,process, nodejs还提供了很多其他十分有用的API :
nodejs的框架
除了基本上的nodejs之外,nodejs还有十分多优秀的框架,借助这些框架他们能是nodejs流程的搭建更加容易和强大。
像AdonisJs,express,koa,http://Socket.io等等。
本文作者:flydean流程那些事
本文链接:http://www.flydean.com/nodejs-kickoff/
本文来源:flydean的博客