飞道的博客

node学习(一)基本使用

190人阅读  评论(0)

node学习(一)node的基本使用

Node是一个基于Chrome V8引擎的运行环境,使用了一个事件驱动、非阻塞式I/O模型,让JavaScript 运行在服务端的开发平台。

1. 原理初探

学一个框架时,弄清楚其原理是十分重要的,因为是初学,所以这里只给出了node的大体结构,深入探究将在最后几章给出。

1.1 node架构:

可以看到,除了node自己提供的标准库之外,node的底层操作主要由以下模块支持:

  • Node bindings: 将调用的java方法传递到c++层面执行,执行完毕后回调

  • V8引擎:v8是Google推出的java vm,其提供了javascript在非浏览器端的运行环境,并且十分高效

  • Libuv:其提供了跨平台、线程池、事件池、异步I/O的能力

  • C-ares:提供了异步处理DNS相关的能力

  • http_parser、OpenSSL、zlib等:提供包括 http 解析、SSL、数据压缩等其他的能力。

1.2. 底层交互

如图,我们在 Java 中调用的方法,最终都会通过 process.binding 传递到 C/C++ 层面,最终由他们来执行真正的操作。Node.js 即这样与操作系统进行互动。

1.3. 单线程应对高并发

node的主线程是单线程的,这意味着如果我们使用过多的同步代码,那么主线程中后面的代码就会被阻塞。那么,为什么单线程的node能够处理万级的并发呢?

1.4. 事件驱动/事件循环

为了效率,我们发现在node代码的编写时用到了大量的异步方法,这些方法是如何实现的呢?

1、每个Node.js进程只有一个主线程在执行程序代码,形成一个执行栈execution context stack)。

2、主线程之外,还维护了一个"事件队列"(Event queue)。当用户的网络请求或者其它的异步操作到来时,node都会把它放到Event Queue之中,此时并不会立即执行它,代码也不会被阻塞,继续往下走,直到主线程代码执行完毕。

3、主线程代码执行完毕完成后,然后通过Event Loop,也就是事件循环机制,开始到Event Queue的开头取出第一个事件,从线程池中分配一个线程去执行这个事件,接下来继续取出第二个事件,再从线程池中分配一个线程去执行,然后第三个,第四个。主线程不断的检查事件队列中是否有未执行的事件,直到事件队列中所有事件都执行完了,此后每当有新的事件加入到事件队列中,都会通知主线程按顺序取出交EventLoop处理。当有事件执行完毕后,会通知主线程,主线程执行回调,线程归还给线程池。

4、主线程不断重复上面的第三步。

2. 基本使用

nodejs中的语法和javascript一致,所以不再赘述,直接上模块使用:

2.1 fs文件系统

首先引入fs模块

const fs = require('fs');
2.1.1同步执行
try {
   
  consta file = fs.readFileSync('文件');
  console.log('读取成功');
} catch (err) {
   
  // 处理错误
}

同步方法会阻塞主线程

2.1.2.异步执行
fs.readFile('文件', (err, data) => {
   
  if (err) throw err;
  console.log(data); // 回调读取数据
});

若要使得某一个操作在异步方法之后执行,有两种操作:

  1. 在异步方法的回调中执行
  2. 使用promise
2.1.3 promise
const fs = require('fs/promises');

(async function(from, to) {
   
  try {
   
    await fs.rename(from, to);
    const stats = await fs.stat(to);
    console.log(`文件属性: ${
     JSON.stringify(stats)}`);
  } catch (error) {
   
    console.error('出错:', error.message);
  }
})('旧文件', '新文件');

2.2 http

node内置http模块,但是为了支持所有可能的 HTTP 应用程序,Node.js 的 HTTP API 都是非常底层的。 它仅进行流处理和消息解析。 它将消息解析为消息头和消息主体,但不会解析具体的消息头或消息主体。

const http = require('http')

创建http服务:

http.createServer((req, res) => {
   
    res.end('hello world');
}, 3000);// 监听3000端口

可以看到,

  1. 该服务会收发所有的请求,若要针对不同的请求路径,则需要不断判断req.path 来进行处理,较为繁琐

  2. 返回的页面不支持添加参数,所以只能返回静态页面

针对这两个问题,我们需要用到第三方的模块来进行简化

  1. 使用express、kora等框架收发请求
  2. 使用ejs、jade等模板引擎来渲染页面

转载:https://blog.csdn.net/lzr_ps/article/details/116758819
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场