@charset "UTF-8";
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.hljs-comment, .hljs-quote { color: rgba(212, 208, 171, 1) }
.hljs-deletion, .hljs-name, .hljs-regexp, .hljs-selector-class, .hljs-selector-id, .hljs-tag, .hljs-template-variable, .hljs-variable { color: rgba(255, 160, 122, 1) }
.hljs-built_in, .hljs-builtin-name, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-type { color: rgba(245, 171, 53, 1) }
.hljs-attribute { color: rgba(255, 215, 0, 1) }
.hljs-addition, .hljs-bullet, .hljs-string, .hljs-symbol { color: rgba(171, 227, 56, 1) }
.hljs-section, .hljs-title { color: rgba(0, 224, 224, 1) }
.hljs-keyword, .hljs-selector-tag { color: rgba(220, 198, 224, 1) }
.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(43, 43, 43, 1); color: rgba(248, 248, 242, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
@media screen and (-ms-high-contrast: active) { .hljs-addition, .hljs-attribute, .hljs-built_in, .hljs-builtin-name, .hljs-bullet, .hljs-comment, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-quote, .hljs-string, .hljs-symbol, .hljs-type { color: rgba(181, 213, 255, 1) } .hljs-keyword, .hljs-selector-tag { font-weight: 700 } }

一、什么是Node.js

Node.js是基于Chrome V8引擎构建的运行在服务端的JavaScript运行环境,可以让JavaScript脱离浏览器运行于本地电脑或者服务器端。

它拥有非阻塞I/O和事件驱动等优秀特性,使得它在高并发、I/O密集等场景下能够发挥出极高的性能,而它的NPM(Node.js Package Manager)包管理器也为开发人员提供了极大的方便。

二、Node.js的特点

  1. 异步I/O:Node.js采用了异步非阻塞I/O模型,使得程序在进行IO操作时不会被阻塞,大大增加了程序的吞吐量。

  2. 事件驱动:Node.js采用事件驱动的编程模型,将每一个I/O操作视为一个事件,通过事件监听和回调函数的方式来处理事件,实现异步编程。

  3. 单线程:Node.js采用了单线程的模型,使得在处理I/O密集型任务时效率非常高,但在处理计算密集型任务时性能不如多线程。

  4. 跨平台:Node.js是基于Javascript的,具有良好的跨平台特性,可以在Windows、Linux、Mac等多个不同平台上运行。

  5. 高性能:Node.js采用了V8引擎和非阻塞I/O等特性,使得它在高并发和I/O密集的场景下表现出了出色的性能。

三、Node.js的事件驱动编程模型

Node.js采用事件驱动的编程模型,将每一个I/O操作视为一个事件,通过事件监听和回调函数的方式来处理事件,实现异步编程。

以下是一个简单的事件驱动的例子:

//开启一个服务,并监听端口号为8000
var http = require('http');
var server = http.createServer().listen(8000); //当有请求时触发
server.on('request', function (req, res) {
res.writeHead(200, {'Content-Type': 'text/plain'});
res.write('Hello World!');
res.end();
}); //服务器启动时触发
server.on('listening', function () {
console.log('Server started!');
});

代码中使用http.createServer()创建了一个http server对象,并监听了端口号为8000,当有请求发送过来时就会触发request事件,可以通过server.on()方法来监听事件,并通过回调函数来处理事件。

四、Node.js的异步IO机制

Node.js采用了非阻塞I/O模型,当进行I/O操作时,程序会继续执行下一条指令,而不会等待I/O操作的完成。当I/O操作完成后,Node.js会通过事件驱动的方式,将完成事件推到事件队列并进行处理。

以下是一个简单的异步I/O的例子:

var fs = require('fs');

//异步读取文件
fs.readFile('test.txt', function (err, data) {
if (err) {
console.log(err);
}
console.log(data.toString());
}); console.log('程序执行完毕');

代码中使用fs.readFile()方法异步读取文件,当读取完成后通过回调函数来处理结果。程序会先输出程序执行完毕,然后再输出读取到的文件内容。

五、Node.js的模块机制

Node.js采用了CommonJS规范来管理模块和依赖,每一个文件表示一个模块,exports对象用来对外暴露模块的方法、属性等。可以通过require()函数来引入其他模块。

以下是一个简单的模块导出和引入的例子:

//module1.js
var name = 'module1'; function sayHello() {
console.log('Hello ' + name);
}; //将sayHello方法导出
module.exports.sayHello = sayHello; //module2.js
var module1 = require('./module1');
module1.sayHello();

在module1.js中,将sayHello方法通过module.exports对象进行了导出。在module2.js中,使用require()方法来引入module1.js模块,并通过module1对象调用sayHello方法。

六、Node.js的事件循环机制

Node.js的事件循环机制采用类似浏览器的事件循环机制,即将任务加入到事件队列中,在事件循环器中不断取出一个事件,并进行处理。

以下是事件循环机制的代码示例:

//异步操作1
setTimeout(function () {
console.log('异步操作1');
}, 1000); //异步操作2
setTimeout(function () {
console.log('异步操作2');
}, 500); //同步操作
console.log('同步操作');

代码中分别有两个异步操作和一个同步操作。在执行到setTimeout()方法时,Node.js会将这个操作加入到事件队列中,并在等待指定的时间后将其推入事件循环器中。在等待的过程中不会被阻塞,而会继续执行后面的同步操作。当事件循环器从队列中取出异步操作时,就会执行相应的回调函数。

七、使用Node.js进行异步编程的实践

1.使用async/await

async/await是ES7中引入的异步编程新特性,可以通过简单的语法实现异步编程。

以下是一个使用async/await编写的异步读取文件的例子:

var fs = require('fs');

async function readFile() {
try {
var data = await fs.promises.readFile('test.txt', 'utf8');
console.log(`读取成功:${data}`);
} catch (err) {
console.log(err);
}
} readFile();

代码中通过async定义一个异步函数,使用await来等待异步读取文件的操作完成。在try-catch语句中可以处理读取文件时产生的错误。

2.使用Promise

Promise是一种基于回调函数的异步编程解决方案,可以通过链式调用和.then()方法来实现异步编程。

以下是一个使用Promise实现异步读取文件的例子:

var fs = require('fs');

function readFile() {
return new Promise(function (resolve, reject) {
fs.readFile('test.txt', 'utf8', function (err, data) {
if (err) {
reject(err);
}
resolve(data);
});
});
} readFile().then(function (data) {
console.log(`读取成功:${data}`);
}).catch(function (err) {
console.log(err);
});

代码中定义了一个readFile方法,使用Promise包装异步读取文件的操作。在.then()方法中可以处理读取文件成功时的情况,在.catch()方法中处理读取文件失败时的情况。

以下是一些Node.js异步编程中常用的API和代码示例说明:

1.异步读写文件

异步读取文件:

var fs = require('fs');
fs.readFile(filepath, options, callback);

异步写入文件:

var fs = require('fs');
fs.writeFile(filepath, data, options, callback);

其中filepath表示文件路径,options表示读取/写入的配置参数,callback为回调函数。

2.异步HTTP请求

使用Node.js可以简单地发起HTTP请求。以下是一个使用request库发起GET请求的例子:

var request = require('request');
request('http://example.com', function (error, response, body) {
if (!error && response.statusCode == 200) {
console.log(body);
}
});

3.异步操作Promise

Promise是一种基于回调函数的异步编程解决方案,可以通过链式调用和.then()方法来实现异步编程。以下是一个使用fetch库取得API数据的例子:

fetch(url)
.then(response => {
if (response.ok) {
return response.json();
} else {
throw new Error('网络错误');
}
})
.then(data => {
console.log(data);
})
.catch(error => {
console.log(error);
});

4.事件驱动编程

Node.js采用事件驱动的编程模型,将每一个I/O操作视为一个事件,通过事件监听和回调函数的方式来处理事件,实现异步编程。以下是一个使用EventEmitter类监听事件的例子:

var EventEmitter = require('events');
var myEmitter = new EventEmitter(); myEmitter.on('event', function (a, b) {
console.log(a + b);
}); myEmitter.emit('event', 'Hello', 'World');

使用EventEmitter类实例化一个事件对象,可以使用.on()方法绑定事件并指定回调函数。使用.emit()方法来触发事件并传递参数给回调函数。

以上是一些常用的Node.js异步编程API和代码示例,可以作为异步编程的参考。

总结

总之,Node.js的异步编程能力是其非常重要的特性和优势,也是Node.js应用广泛的重要原因之一。在Node.js中,采用了事件驱动和异步I/O等机制来实现高性能的异步编程。

在实践中,可以使用async/await、Promise、事件驱动等方式实现Node.js的异步编程。同时也要注意异步操作可能带来的问题,例如回调地狱、内存泄漏等,应该结合具体的应用场景来选择合适的方案进行异步编程。

在开发过程中,不断学习Node.js异步编程的相关知识和实践方法,不断优化和改进程序,才能让Node.js程序发挥出更优秀的性能和效果。

掌握Node.js原理,开启异步编程之旅的更多相关文章

  1. 异步编程之Promise(2):探究原理

    异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2):探究原理 异步编程之Promise(3):拓展进阶 异步编程之Generator(1)--领略魅 ...

  2. 异步编程之Generator(1)——领略魅力

    异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2):探究原理 异步编程之Promise(3):拓展进阶 异步编程之Generator(1)--领略魅 ...

  3. 异步编程之Promise(3):拓展进阶

    异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2):探究原理 异步编程之Promise(3):拓展进阶 异步编程之Generator(1)--领略魅 ...

  4. (翻译)异步编程之Promise(1):初见魅力

    原文:https://www.promisejs.org/ by Forbes Lindesay 异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2) ...

  5. Javascript异步编程之setTimeout与setInterval详解分析(一)

    Javascript异步编程之setTimeout与setInterval 在谈到异步编程时,本人最主要会从以下三个方面来总结异步编程(注意:特别解释:是总结,本人也是菜鸟,所以总结不好的,请各位大牛 ...

  6. 异步编程之co——源码分析

    异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2):探究原理 异步编程之Promise(3):拓展进阶 异步编程之Generator(1)--领略魅 ...

  7. 【转】Javascript异步编程之setTimeout与setInterval

    Javascript异步编程之setTimeout与setInterval 转自:http://www.tuicool.com/articles/Ebueua 在谈到异步编程时,本人最主要会从以下三个 ...

  8. 异步编程之Generator(2)——剖析特性

    异步编程系列教程: (翻译)异步编程之Promise(1)--初见魅力 异步编程之Promise(2):探究原理 异步编程之Promise(3):拓展进阶 异步编程之Generator(1)--领略魅 ...

  9. net异步编程之await

    net异步编程之await 初探asp.net异步编程之await   终于毕业了,也顺利进入一家期望的旅游互联网公司.27号入职.放肆了一个多月没写代码,好方啊. 另外一下观点均主要针对于await ...

  10. 玩node-images模块---Node.js轻量级跨平台图像编解码库

    Node.js轻量级跨平台图像编解码库 github:https://github.com/zhangyuanwei/node-images Features 功能特性 轻量级:无需安装任何图像处理库 ...

随机推荐

  1. ABAQUS阻尼设置

    结构阻尼 瑞利阻尼

  2. 一个ABAQUS model需要的Component

    component of abaqus model Abaqus模型由几个不同的组件组成,它们共同描述了要分析的物理问题. a abaqus model 至少要有: discrete goemtry ...

  3. 关闭 ReportCrash 进程防止CPU占用率过高 [MacBook]

    关闭 ReportCrash 的原因 自己 MacBook Pro 总是过载,机器很热.结果通过看进程 top 命令,看到 ReportCrash 占用了了过高的 CPU,而且好像我用不上.于是乎,得 ...

  4. [Qt基础-07 QSignalMapper]

    QSignalMapper 本文主要根据QT官方帮助文档以及日常使用,简单的介绍一下QSignalMapper的功能以及使用 文章目录 QSignalMapper 简介 使用方法 主要的函数 信号和槽 ...

  5. linux clickhouse 密码设置

    默认密码 clickhouse 安装好之后,系统默认的登录账号密码是 /etc/clickhouse-server/users.d/default-password.xml 文件中配置的,默认密码是 ...

  6. PaddleOCR学习笔记2-初步识别服务

    今天初步实现了网页,上传图片,识别显示结果到页面的服务.后续再完善. 采用flask + paddleocr+ bootstrap快速搭建OCR识别服务. 代码结构如下: 模板页面代码文件如下: up ...

  7. AIX操作系统基本命令

    1,内核 bootinfo -k   2,硬件 bootinfo -r lscfg |grep proc lspv lscfg   3,操作系统 oslevel -r oslevel -s uname ...

  8. javascript for...in

    在JS中我们最常见的循环语句是for循环语句,一个简单的for循环语句如下: for(var i = 0, n = 100; i < n; i++){ // to do somethings . ...

  9. Sentinel源码—2.Context和处理链的初始化

    大纲 1.Sentinel底层的核心概念 2.Sentinel中Context的设计思想与源码实现 3.Java SPI机制的引入 4.Java SPI机制在Sentinel处理链中的应用 5.Sen ...

  10. 机器人操作系统ROS2之安装(Ubuntu 24)

    根据官网说明,ROS2是支持MAC的,本来打算在手头的MAC 15.3.2装一个,虽然要自己编译ROS2系统,但是想着比虚拟机性能好,就兴冲冲的开始了,也没在意官网提示尽量还是装安装版.最后折腾了几天 ...