简单了解 node net 模块
简单了解 node net 模块
文章记录了对net 模块的简单理解分析。
- net模块
- 简单使用
- net.Server 类
- net.Socket 类
- 总结
1.1 net模块
Node.js 的 Net模块用于创建基于流的 TCP 或 IPC 的服务器(net.createServer())与客户端(net.createConnection())。
Node.js 的 Net模块是基于TCP协议的socket网路编程模块。
主要的两个部分
- net.Server 类
- net.Socket 类
TCP是面向连接的传输层协议,重要的特征是建立传输前的三次握手

当两端都确认了,才开始相互传输数据,会话创建过后,服务端和客户端分别提供一个套接字,形成此次的连接。

套接字 socket 是一个 双工流,既可以通过 data事件读取数据另一方发过来的数据,又可以通过write()方法向另一方发送数据。
1.2. 简单使用
先写个小例子:
TCP服务端:
let net = require('net')
let server = net.createServer({}, socket => {
// socket 是一个双工流
console.log('client connected')
socket.on('data', data => {
console.log(data.toString())
socket.write('server: hello server')
})
// 服务器收到客户端发出的关闭请求时,会触发end事件
socket.on('end', () => {
console.log('client disconnected')
})
socket.on('close', () => {
console.log('client closed')
})
})
server.listen(8080, () => {
console.log('server start');
})
TCP客户端:
let net = require('net')
// new net.Socket() 返回的是一个双工流
let client = new net.Socket()
client.connect(8080, 'localhost', () => {
console.log('connected server')
client.write('client: hello server');
})
client.on('data', function (data) {
console.log(data.toString())
});
setTimeout(()=>{
client.end()
},5000)
上面流程调试代码画个简图:(假设此时有两个客户端连接)

TCP服务是以connection 为单位进行服务的。
每一次连接都 会创建一个socket 实例
1.3. net.Server 类
createServer就是一个语法糖,帮助new生成server实例,
server 对象继承了EventEmitter对象 。
this.on('connection', connectionListener); // 此处订阅了connection,等待有连接的时候,发布connecttionListener就会被调用
截取部分代码如下
function Server(options, connectionListener) {
if (!(this instanceof Server))
return new Server(options, connectionListener);
// 调用 EventEmitter 获得属性
EventEmitter.call(this);
// 订阅connection 事件
this.on('connection', connectionListener);
// 统计连接数量
this._connections = 0;
this[async_id_symbol] = -1;
// 会挂载TCP对象
this._handle = null;
this._usingWorkers = false;
this._workers = [];
this._unref = false;
this.allowHalfOpen = options.allowHalfOpen || false;
this.pauseOnConnect = !!options.pauseOnConnect;
}
每当有客户端连接时,就会调用callback函数onconnection函数,创建socket实例,将由c++部分的TCP、Pipe类创建的实例clientHandle挂载在_handle属性上。
然后发布connection事件传递socket 实例
所以server 对象更多的是对socket连接的管理。截取代码如下
function onconnection(err, clientHandle) {
const handle = this;
const self = handle[owner_symbol];
// 超出最大链接数 不让客户端连接
if (self.maxConnections && self._connections >= self.maxConnections) {
clientHandle.close();
return;
}
const socket = new Socket({
handle: clientHandle,
allowHalfOpen: self.allowHalfOpen,
pauseOnCreate: self.pauseOnConnect,
readable: true,
writable: true
});
// 连接数增加
self._connections++;
socket.server = self;
socket._server = self;
DTRACE_NET_SERVER_CONNECTION(socket);
// 发布 connection
self.emit('connection', socket);
}
clientHandle对象如下,它是一个TCP 实例:

onconnection执行的目的是对应用程序构的连接造出一个socket的实例。
self.emit('connection', socket)// 发布 connecttion 事件,传递socket
执行connectionListener (socket),然后就基于此socket实例完成面向connection的数据流读取操作。
补充
如果.createServer 没有传递connectionListener 函数
也可以使用 server.on('connection',connectionListener) // .createServer 传递connectionListener 函数更像语法糖
1.4. net.Socket对象
socket(通过 socket=new Socket() ) 是个双工流源码截取如下:

socket._handle上挂载的实例对象是由C++中的Pipe、TCP实现,大概截取代码如下
//<------Socket函数----->
this._handle = createHandle(fd, false); //<------createHandle函数----->
function createHandle(fd, is_server) {
validateInt32(fd, 'fd', 0);
const type = guessHandleType(fd);
if (type === 'PIPE') {
return new Pipe(
is_server ? PipeConstants.SERVER : PipeConstants.SOCKET
);
} if (type === 'TCP') {
return new TCP(
is_server ? TCPConstants.SERVER : TCPConstants.SOCKET
);
} throw new ERR_INVALID_FD_TYPE(type);
}
由c++部分的TCP、Pipe类创建由internalBinding('tcp_wrap')导出的TCP对象,如下:

Node.js 的 Net模块是对 c++部分的TCP、Pipe类创建的socket 进行了抽象封装。截取个代码如下

所以socke实例有Writable,Readable,和TCP的相关功能函数。
1.5. 总结
TCP服务是一connection 为单位进行服务的。
理解net模块可以多看文档,代码实践。
还可以看看TCP协议,链接建立握手,慢启动拥塞控制,Nagle算法解决的问题等等。

文章记录了对net 模块的简单理解分析。例子用词比较粗糙,理解不准确之处,还请教正。
简单了解 node net 模块的更多相关文章
- node.js学习(三)简单的node程序&&模块简单使用&&commonJS规范&&深入理解模块原理
一.一个简单的node程序 1.新建一个txt文件 2.修改后缀 修改之后会弹出这个,点击"是" 3.运行test.js 源文件 使用node.js运行之后的. 如果该路径下没有该 ...
- 使用node-gyp编写简单的node原生模块
通过样例,让我们了解如何编写一个node的原生模块.当然,这篇文章还有一个目的,是为了方便以后编写关于node-gyp的文章,搭建初始环境. 基于node-addon-api 基于node-addon ...
- 简单了解 node http(一)
简单了解 node http 模块 文章记录了对http 模块的简单使用与理解. http 服务端 http 客户端 总结 1. http 服务端 先写个小例子 服务端: let http = req ...
- 深入浅出node(2) 模块机制
这部分主要总结深入浅出Node.js的第二章 一)CommonJs 1.1CommonJs模块定义 二)Node的模块实现 2.1模块分类 2.2 路径分析和文件定位 2.2.1 路径分析 2.2.2 ...
- 如何发布一个自定义Node.js模块到NPM(详细步骤)
咱们闲话不多说,直接开始! 由于我从没有使用过MAC,所以我不保证本文中介绍的操作与MAC一致. 文章开始我先假定各位已经在window全局安装了Node.js,下面开始进行详细步骤介绍: 本文本着, ...
- Node.js 模块
稳定性: 5 - 锁定 Node 有简单的模块加载系统.在 Node 里,文件和模块是一一对应的.下面例子里,foo.js 加载同一个文件夹里的 circle.js 模块. foo.js 内容: va ...
- Node文件模块
在上一篇文章中有提到,Node模块分为核心模块和文件模块,接下来就简单总结一下文件模块. 文件模块则是在运行时动态加载,需要完整的路径分析.文件定位.编译执行过程.速度相比核心模块稍微慢一些,但是用的 ...
- node基础—模块系统
模块的概念 为了让Node.js的文件可以相互调用,Node.js提供了一个简单的模块加载系统. 在 Node.js 中,文件和模块是一一对应的(每个文件被视为一个独立的模块),换言之,一个 Node ...
- Node.js模块定义总结
为了让Node.js的文件可以相互调用,Node.js提供了一个简单的模块系统.模块是Node.js 应用程序的基本组成部分,文件和模块是一一对应的.换言之,一个 Node.js 文件就是一个模块,这 ...
随机推荐
- STS(Spring tool suite)集成配置jdk,maven和Tomcat
STS是spring官网的一个集成开发工具,最近在学springboot,所以用了. 在本文中将介绍如下内容: 搭建开发的 JDK(1.8) 环境. 配置 Maven 环境. 配置 Tomcat 环境 ...
- WebView的基础用法
新建一个WebView项目,然后修改activity_main.xml布局文件中的代码,如下所示: <LinearLayout xmlns:android="http://schema ...
- shiro.ini
# # Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreeme ...
- ffmpeg剪切视频
测试的时候需要用到视频,原片太大了,就剪切几分钟来测试 ffmpeg -i input.mp4 -ss 0 -t 300 -acodec copy -vcodec copy -scodec copy ...
- JAVA-Runnable、Callable、FutureTask
通常,创建线程的执行单元有两种,一种是直接继承 Thread,另外一种就是实现 Runnable 接口. 但这两种都有一个问题就是无法有返回值,且子线程在执行过程中无法抛出异常.想线程有返回值,可以使 ...
- python自然语言处理学习笔记1
1.搭建环境 下载anaconda并安装,(其自带python2.7和一些常用包,NumPy,Matplotlib),第一次启动使用spyder 2.下载nltk import nltk nltk.d ...
- jquery绝对路径
<strong>1.修改为绝对路径</strong> $(document).ready(function(){ $.get("${pageContext.reque ...
- PRISM 4 - RegisterViewWithRegion & Custom Export Attributes
5down votefavorite I am using Prism 4 with MEF Extensions and the MVVM pattern. During initializat ...
- [Nova ERROR] InternalError: Nova requires QEMU version 2.5.0 or greater.
目录 文章目录 目录 问题 调查 解决 问题 nova-compute service 启动失败 InternalError: Nova requires QEMU version 2.5.0 or ...
- Nginx搭建动态静态服务器
Nginx做静态资源服务器优于Tomcat 区分静态资源,动态资源请求 使用域名区分! 如果是动态资源请求 反向代理到 Tomcat 如果 是静态资源请求 直接走本地Nginx 配置: ###静态 ...