NodeJS学习笔记 (10)网络TCP-net(ok)
模块概览
net模块是同样是nodejs的核心模块。在http模块概览里提到,http.Server继承了net.Server,此外,http客户端与http服务端的通信均依赖于socket(net.Socket)。也就是说,做node服务端编程,net基本是绕不开的一个模块。
从组成来看,net模块主要包含两部分,了解socket编程的同学应该比较熟悉了:
- net.Server:TCP server,内部通过socket来实现与客户端的通信。
- net.Socket:tcp/本地 socket的node版实现,它实现了全双工的stream接口。
本文从一个简单的 tcp服务端/客户端 的例子开始讲解,好让读者有个概要的认识。接着再分别介绍 net.Server、net.Socket 比较重要的API、属性、事件。
对于初学者,建议把文中的例子本地跑一遍加深理解。
简单的 server+client 例子
tcp服务端程序如下:
var net = require('net'); var PORT = 3000;
var HOST = '127.0.0.1'; // tcp服务端
var server = net.createServer(function(socket){
console.log('服务端:收到来自客户端的请求'); socket.on('data', function(data){
console.log('服务端:收到客户端数据,内容为{'+ data +'}'); // 给客户端返回数据
socket.write('你好,我是服务端');
}); socket.on('close', function(){
console.log('服务端:客户端连接断开');
});
});
server.listen(PORT, HOST, function(){
console.log('服务端:开始监听来自客户端的请求');
});
tcp客户端如下:
var net = require('net'); var PORT = 3000;
var HOST = '127.0.0.1'; // tcp客户端
var client = net.createConnection(PORT, HOST); client.on('connect', function(){
console.log('客户端:已经与服务端建立连接');
}); client.on('data', function(data){
console.log('客户端:收到服务端数据,内容为{'+ data +'}');
}); client.on('close', function(data){
console.log('客户端:连接断开');
}); client.end('你好,我是客户端');
运行服务端、客户端代码,控制台分别输出如下:
服务端:
服务端:开始监听来自客户端的请求
服务端:收到来自客户端的请求
服务端:收到客户端数据,内容为{你好,我是客户端}
服务端:客户端连接断开
客户端:
客户端:已经与服务端建立连接
客户端:收到服务端数据,内容为{你好,我是服务端}
客户端:连接断开
服务端 net.Server
server.address()
返回服务端的地址信息,比如绑定的ip地址、端口等。
console.log( server.address() );
// 输出如下 { port: 3000, family: 'IPv4', address: '127.0.0.1' }
server.close(callback])
关闭服务器,停止接收新的客户端请求。有几点注意事项:
- 对正在处理中的客户端请求,服务器会等待它们处理完(或超时),然后再正式关闭。
- 正常关闭的同时,callback 会被执行,同时会触发 close 事件。
- 异常关闭的同时,callback 也会执行,同时将对应的 error 作为参数传入。(比如还没调用 server.listen(port) 之前,就调用了server.close())
下面会通过两个具体的例子进行对比,先把结论列出来
- 已调用server.listen():正常关闭,close事件触发,然后callback执行,error参数为undefined
- 未调用server.listen():异常关闭,close事件触发,然后callback执行,error为具体的错误信息。(注意,error 事件没有触发)
例子1:服务端正常关闭
var net = require('net');
var PORT = 3000;
var HOST = '127.0.0.1';
var noop = function(){}; // tcp服务端
var server = net.createServer(noop); server.listen(PORT, HOST, function(){ server.close(function(error){
if(error){
console.log( 'close回调:服务端异常:' + error.message );
}else{
console.log( 'close回调:服务端正常关闭' );
}
});
}); server.on('close', function(){
console.log( 'close事件:服务端关闭' );
}); server.on('error', function(error){
console.log( 'error事件:服务端异常:' + error.message );
});
输出为:
close事件:服务端关闭
close回调:服务端正常关闭
例子2:服务端异常关闭
代码如下
var net = require('net');
var PORT = 3000;
var HOST = '127.0.0.1';
var noop = function(){}; // tcp服务端
var server = net.createServer(noop); // 没有正式启动请求监听
// server.listen(PORT, HOST); server.on('close', function(){
console.log( 'close事件:服务端关闭' );
}); server.on('error', function(error){
console.log( 'error事件:服务端异常:' + error.message );
}); server.close(function(error){
if(error){
console.log( 'close回调:服务端异常:' + error.message );
}else{
console.log( 'close回调:服务端正常关闭' );
}
});
输出为:
close事件:服务端关闭
close回调:服务端异常:Not running
server.ref()/server.unref()
了解node事件循环的同学对这两个API应该不陌生,主要用于将server 加入事件循环/从事件循环里面剔除,影响就在于会不会影响进程的退出。
对出学习net的同学来说,并不需要特别关注,感兴趣的自己做下实验就好。
事件 listening/connection/close/error
- listening:调用 server.listen(),正式开始监听请求的时候触发。
- connection:当有新的请求进来时触发,参数为请求相关的 socket。
- close:服务端关闭的时候触发。
- error:服务出错的时候触发,比如监听了已经被占用的端口。
几个事件都比较简单,这里仅举个 connection 的例子。
从测试结果可以看出,有新的客户端连接产生时,net.createServer(callback) 中的callback回调 会被调用,同时 connection 事件注册的回调函数也会被调用。
事实上,net.createServer(callback) 中的 callback 在node内部实现中 也是加入了做为 connection事件 的监听函数。感兴趣的可以看下node的源码。
var net = require('net');
var PORT = 3000;
var HOST = '127.0.0.1';
var noop = function(){}; // tcp服务端
var server = net.createServer(function(socket){
socket.write('1. connection 触发\n');
}); server.on('connection', function(socket){
socket.end('2. connection 触发\n');
}); server.listen(PORT, HOST);
通过下面命令测试下效果
curl http://127.0.0.1:3000
输出:
1. connection 触发
2. connection 触发
客户端 net.Socket
在文章开头已经举过客户端的例子,这里再把例子贴一下。(备注:严格来说不应该把 net.Socket 叫做客户端,这里方便讲解而已)
单从node官方文档来看的话,感觉 net.Socket 比 net.Server 要复杂很多,有更多的API、事件、属性。但实际上,把 net.Socket 相关的API、事件、属性 进行归类下,会发现,其实也不是特别复杂。
具体请看下一小节内容。
var net = require('net'); var PORT = 3000;
var HOST = '127.0.0.1'; // tcp客户端
var client = net.createConnection(PORT, HOST); client.on('connect', function(){
console.log('客户端:已经与服务端建立连接');
}); client.on('data', function(data){
console.log('客户端:收到服务端数据,内容为{'+ data +'}');
}); client.on('close', function(data){
console.log('客户端:连接断开');
}); client.end('你好,我是客户端');
API、属性归类
以下对net.Socket的API跟属性,按照用途进行了大致的分类,方便读者更好的理解。大部分API跟属性都比较简单,看下文档就知道做什么的,这里就先不展开。
连接相关
- socket.connect():有3种不同的参数,用于不同的场景;
- socket.setTimeout():用来进行连接超时设置。
- socket.setKeepAlive():用来设置长连接。
- socket.destroy()、socket.destroyed:当错误发生时,用来销毁socket,确保这个socket上不会再有其他的IO操作。
数据读、写相关
socket.write()、socket.end()、socket.pause()、socket.resume()、socket.setEncoding()、socket.setNoDelay()
数据属性相关
socket.bufferSize、socket.bytesRead、socket.bytesWritten
事件循环相关
socket.ref()、socket.unref()
地址相关
- socket.address()
- socket.remoteAddress、socket.remoteFamily、socket.remotePort
- socket.localAddress/socket.localPort
事件简介
- data:当收到另一侧传来的数据时触发。
- connect:当连接建立时触发。
- close:连接断开时触发。如果是因为传输错误导致的连接断开,则参数为error。
- end:当连接另一侧发送了 FIN 包的时候触发(读者可以回顾下HTTP如何断开连接的)。默认情况下(allowHalfOpen == false),socket会完成自我销毁操作。但你也可以把 allowHalfOpen 设置为 true,这样就可以继续往socket里写数据。当然,最后你需要手动调用 socket.end()
- error:当有错误发生时,就会触发,参数为error。(官方文档基本一句话带过,不过考虑到出错的可能太多,也可以理解)
- timeout:提示用户,socket 已经超时,需要手动关闭连接。
- drain:当写缓存空了的时候触发。(不是很好描述,具体可以看下stream的介绍)
- lookup:域名解析完成时触发。
相关链接
官方文档: https://nodejs.org/api/net.html#net_socket_destroy_exception
NodeJS学习笔记 (10)网络TCP-net(ok)的更多相关文章
- nodejs学习笔记之网络编程
了解一下OSI七层模型 OSI层 功能 TCP/IP协议 应用层 文件传输,电子邮件,文件服务,虚拟终端 TFTP,HTTP,SNMP,FTP,SMTP,DNS,Telnet 表示层 数据格式化 ...
- python学习笔记10 ----网络编程
网络编程 网络编程需要知道的概念 网络体系结构就是使用这些用不同媒介连接起来的不同设备和网络系统在不同的应用环境下实现互操作性,并满足各种业务需求的一种粘合剂.网络体系结构解决互质性问题彩是分层方法. ...
- NodeJS学习笔记 (8)网络服务-http-server(ok)
http服务端概览 创建server 几行代码搞定 var http = require('http'); var requestListener = function(req, res){ res. ...
- NodeJS学习笔记 (9)网络服务-https(ok)
模块概览 这个模块的重要性,基本不用强调了.在网络安全问题日益严峻的今天,网站采用HTTPS是个必然的趋势. 在nodejs中,提供了 https 这个模块来完成 HTTPS 相关功能.从官方文档来看 ...
- NodeJS学习笔记 (5)网络服务-http-req(ok)
原文:https://github.com/chyingp/nodejs-learning-guide 自己敲代码: 概览 本文的重点会放在req这个对象上.前面已经提到,它其实是http.Incom ...
- NodeJS学习笔记 (4)网络服务-http(ok)
原文:https://github.com/chyingp/nodejs-learning-guide 自己敲代码: http模块概览 大多数nodejs开发者都是冲着开发web server的目的选 ...
- NodeJS学习笔记 (7)网络服务-http-client(ok)
原文:https://github.com/chyingp/nodejs-learning-guide 自己敲代码: ClientRequest概览 当你调用 http.request(options ...
- NodeJS学习笔记 (6)网络服务-http-res(ok)
原文:https://github.com/chyingp/nodejs-learning-guide 自己敲代码: 概览 http模块四剑客之一的res,应该都不陌生了.一个web服务程序,接受到来 ...
- NodeJS学习笔记 (12)网络地址解析-url(ok)
模块概述 nodejs中,提供了url这个非常实用的模块,用来做URL的解析.在做node服务端的开发时会经常用到.使用很简单,总共只有3个方法. 正式讲解前,各位同学先把下面这个图记在心上(来自no ...
随机推荐
- 安装Oracle RAC 11g
1.Oracle Enterprise Linux 和 iSCSI 上构建 Oracle RAC 11g 集群 2.Oracle RAC 的所有共享磁盘存储将基于 iSCSI,iSCSI 使用在第三个 ...
- xBIM 基础16 IFC的空间层次结构
系列目录 [已更新最新开发文章,点击查看详细] 本篇介绍如何从文件中检索空间结构.IFC中的空间结构表示层次结构的嵌套结构,表示项目,站点,建筑物,楼层和空间.如果您查看IFC文档, 您会发现 ...
- 编程语言与Python学习(二)
1.1 流程控制之for循环 1 迭代式循环:for,语法如下 for i in range(10): 缩进的代码块 2 break与continue(同上) 3 循环嵌套 for i in rang ...
- Linux部署之批量自动安装系统之测试篇
1. 客户端从网络启动如下 2. 复制vesamenu.c32文件可解决上面的问题 3. 客户端再次启动 4. 选择第一个进 ...
- [HNOI2008]越狱 快速幂 逆推
考虑越狱的情况有些复杂,不如考虑总情况减去不越狱的情况. 显然,总情况为 $m^n$ 种,不越狱的情况为 $m*(m-1)*(m-1)*(m-1)....$ 即为 $m*(m-1)^(n-1)$. 做 ...
- vue的表格加单选框
https://www.cnblogs.com/calamus/p/8569196.html
- C语言基础 (1) 操作系统介绍,linux入门
第一天 一.操作系统 1.1.1操作系统的目标 ·方便:使计算机系统易于使用 ·有效:以更有效的方式使用计算机系统资源 ·扩展:方便用户有效开发.测试和引进新功能 1.1.2 操作系统的地位 操作系统 ...
- HDU-2222 Keywords Search 字符串问题 AC自动机
题目链接:https://cn.vjudge.net/problem/HDU-2222 题意 给一些关键词,和一个待查询的字符串 问这个字符串里包含多少种关键词 思路 AC自动机模版题咯 注意一般情况 ...
- salt 安装kubernetes集群3节点
[root@linux-node1 k8s]# tree .├── etcd.sls├── files│ ├── cfssl-1.2│ │ ├── cfssl-certinfo_linux ...
- 配置oh-my-zsh
1. 当使用zsh进入庞大的git工程目录下时,会发生cd命令很慢的情况 可以把~/.oh-my-zsh/lib/git.zsh里面的git_prompt_info函数替换为 function git ...