Node.js之网络通讯模块实现浅析


Posted in Javascript onApril 01, 2017

前言

想必我们在用Node.js用的最多的应该是创建http服务,所以对于每个Web开发工程师而言,Node.js的网络相关模块学习是必不可少。

Node.js的网络模块架构

在Node.js的模块里面,与网络相关的模块有Net、DNS、HTTP、TLS/SSL、HTTPS、UDP/Datagram,除此之外,还有v8底层相关的网络模块有tcp_wrap.ccudp_wrap.ccpipe_wrap.ccstream_wrap.cc等等,在Javascript层以及C++层之间通过process.binding进行桥接相互通信。

Node.js之网络通讯模块实现浅析

Net模块

Net模块提供了一些用于底层的网络通信接口,包括创建服务器以及客户端,其中HTTP模块也是基于Net模型的上层封装,在Net模块里面主要提供net.Server以及net.Socket

创建TCP服务端

创建一个TCP服务器,可以通过使用构造函数new net.Server或者使用工厂方法net.createServer,这两个方法都会返回一个net.Server类,可接收两个可选参数。

var net = require('net');

var server = net.createServer(function(socket){

  socket
    .on('data',function(data){
      console.log('socket data',data.toString());
      socket.write( data.toString() );
    })
    .on('end',function(){
      console.log('socket end')
    })
    .on('error',function(error){
      console.log('socket error',error);
    });
});

server.listen(56200,function(){
  console.log('server run at ',server.address());
});

server.on('error',function(err){
  throw err;
});
// 执行后:server run at { address: '::', family: 'IPv6', port: 56200 }

在listen监听的时候没有指定端口的话会自动随意监听一个端口,创建完成一个TCP服务器后,使用tenlent 0.0.0.0 56200,链接后可与服务器进行数据通信。通过createServer实例化一个服务后,服务会去监听客户端请求,与客户端建立了链接之后会在回调里面抛出建链的net.Socket对象。

创建TCP客户端

创建一个TCP客户端链接可以使用构造函数new net.Socket或者其工厂方法net.createConnection,创建成功后都会返回一个net.Socket实例。

var net = require('net');

var client = net.createConnection({port:56200,host:'localhost'});

client.on('connect',function(){
  console.log('client connect');
});

client.on('data',function(data){
  console.log('client data',toString());
});

client.on('error',function(error){
  throw error;
});

client.on('close',function(){
  console.log('client close');
});

Socket

socket是啥这里就不做详细的阐述了,下面主要了解下net.Socket这个构造体主要有提供一些什么方法、监听事件的使用。

相关事件

  1. connect : 当客户端与服务端成功建立链接之后触发,如果链接不上服务器直接抛出error事件错误然后退出node进程。
  2. data : 当客户端收到服务器传送过来的数据或者是客户端传送给服务器的数据的时候触发回调。
  3. end : 当另外一侧发送FIN包断开的时候触发,默认情况下面 (allowHalfOpen == false)socket会自我销毁(如果写入待处理队列里面还没正式响应回包),但是我们可以设置allowHalfOpen参数为true,这样可以继续往该socket里面写数据,但是我们需要自己去调用 end 方法去消耗这个socket,不然可能会造成句柄泄漏。
  4. close : 链接断开的时候触发,但是如果在传输的过程中有错误的话这里会在回调函数里面抛出 error。
  5. timeout : 当socket超时空闲的时候触发,如果要在队列里面销毁需要手动去调close方法。
  6. lookup : 域名解析完成的时候触发。
  7. drain : 写完缓存的时候触发,可使用在上传大小限制中。

相关方法

  1. write() : 服务端给客户端发送数据或者是客户端给服务端发送数据。
  2. address() : 获取服务绑定的socket的IP地址,返回对象有三个属性,分别为端口、host以
  3. 及IPvX版本。
  4. end() : 半关闭socket,会发送一个FIN包,服务器仍然可能发送一些数据,也可以这样调用socket.end(data,encoding)。
  5. pause() : 暂停读取数据,可以用作对数据上传限制。
  6. resume() : 继续数据读取。
  7. setEncoding() : 设置数据流的获取格式。
  8. setKeepAlive() : 允许/禁止keep-alive功能。
  9. setNoDelay() : 禁止Nagele算法,TCP链接默认使用Nagle算法,它们在发送之前数据会被缓存。这是为true的话在每次socket.write()的时候会立即发送数据,默认为true。
  10. setTimeout() : 当一个空闲的socket在多少秒后不活跃会被接受到timeout事件,但是该socket不会停止销毁,需要手动调用end()或者destroy()。表示禁止空闲超时。

相关属性

  1. bufferSize : 当前缓存的等待被发送的字符串的数量。
  2. bytesRead : 收到的字节的数量。
  3. bytesWritten : 发送的字节的数量
  4. destroyed : 标识链接是否已经被破坏,一旦被破环,就不用使用该链接来传输数据。
  5. localAddress : 远程客户端链接本地地址的host。如果我们监听服务的host是0.0.0.0,而客户端链接的是'192.168.1.1',最后的值是后者。
  6. localPort : 本地的端口。
  7. remoteAddress : 客户端IP,如果socket已经是destryed的话,该值为undefined
  8. remoteFamily : 客户端是IPvX

回包异常处理

服务器从客户端接受到需要处理的数据后进入处理环节,再业务逻辑处理完成之前如果socket以外断开的话,待服务器再给客户端回报的时候会直接响应error事件并报错Error : This socket has benn ended by the other part,所以在回报之前服务端需要先判断该socket是否被销毁,如果没有被销毁则回包,如果已经断开则销毁:

var net = require('net');
var biz = require('./biz');
var server = net.createServer(function(socket){

  socket
    .on('data',function(data){
      biz.do(data)
        .then(function(){
          if( !socket.destroyed ) {
            socket.write( data.toString() );
          } else {
            // do some report
            socket.destry();
          }
        })
        .catch(function(){
          !socket.destroyed && socket.end('server handler error');
        });
      
    })
    .on('end',function(){
      console.log('socket end')
    })
    .on('error',function(error){
      console.log('socket error',error);
    });
});

server.listen(56200,function(){
  console.log('server run at ',server.address());
});

server.on('error',function(err){
  throw err;
});

限制客户端数据大小

对请求大小限制是服务安全里面比不可少的一个环节,服务端不能无限大小的去接受客户端发送过来的所有数据,而限制大小就是第一道门槛。

var net = require('net');
var MAX_REQUEST_BYTES = 2 * 1024 * 1024; // 2M
var server = net.createServer(function(socket){

  socket
    .on('data',function(data){
      
      if(data.bytesRead > MAX_REQUEST_BYTES) {
        socket.pause();
        socket.end('data is too big, forbidden');
        // do some report
      }
    })
    .on('end',function(){
      console.log('socket end')
    })
    .on('error',function(error){
      console.log('socket error',error);
    });
});

server.listen(56200,function(){
  console.log('server run at ',server.address());
});

server.on('error',function(err){
  throw err;
});

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript接口实现代码 (Interfaces In JavaScript)
Jun 11 Javascript
window.location.reload()方法刷新页面弹出要再次显示该网页对话框
Apr 24 Javascript
JS通过相同的name进行表格求和代码
Aug 18 Javascript
经过绑定元素时会多次触发mouseover和mouseout事件
Feb 28 Javascript
jQuery 写的简单打字游戏可以提示正确和错误的次数
Jul 01 Javascript
jquery仿苹果的时间/日期选择效果
Mar 08 Javascript
javascript深拷贝的原理与实现方法分析
Apr 10 Javascript
详解用node.js实现简单的反向代理
Jun 26 Javascript
基于vue-cli配置lib-flexible + rem实现移动端自适应
Dec 26 Javascript
Vue.js计算机属性computed和methods方法详解
Oct 12 Javascript
微信小程序实现上传照片代码实例解析
Aug 04 Javascript
vue实现桌面向网页拖动文件的示例代码(可显示图片/音频/视频)
Mar 01 Vue.js
vue-cli+webpack记事本项目创建
Apr 01 #Javascript
JS常见创建类的方法小结【工厂方式,构造器方式,原型方式,联合方式等】
Apr 01 #Javascript
Vue.js实战之Vuex的入门教程
Apr 01 #Javascript
AngularJS页面传参的5种方式
Apr 01 #Javascript
vue2.0构建单页应用最佳实战
Apr 01 #Javascript
vue货币过滤器的实现方法
Apr 01 #Javascript
javascript 中的try catch应用总结
Apr 01 #Javascript
You might like
DedeCms模板安装/制作概述
2007/03/11 PHP
需要使用php模板的朋友必看的很多个顶级PHP模板引擎比较分析
2008/05/26 PHP
PHP自定义序列化接口Serializable用法分析
2017/12/29 PHP
PHP开发api接口安全验证操作实例详解
2020/03/26 PHP
javascript里的条件判断
2007/02/27 Javascript
自己写的Javascript计算时间差函数
2013/10/28 Javascript
JQuery 使用attr方法实现下拉列表选中
2014/10/13 Javascript
javascript数组去重的方法汇总
2015/04/14 Javascript
Java中Timer的用法详解
2015/10/21 Javascript
基于jQuery代码实现圆形菜单展开收缩效果
2017/02/13 Javascript
angular4中关于表单的校验示例
2017/10/16 Javascript
JavaScript适配器模式详解
2017/10/19 Javascript
nodejs实现简单的gulp打包
2017/12/21 NodeJs
JS删除数组里的某个元素方法
2018/02/03 Javascript
对vue2.0中.vue文件页面跳转之.$router.push的用法详解
2018/08/24 Javascript
Vue数据驱动表单渲染,轻松搞定form表单
2019/07/19 Javascript
关于layui 实现点击按钮添加一行(方法渲染创建的table)
2019/09/29 Javascript
vue实现pdf文档在线预览功能
2019/11/26 Javascript
JS实现简单省市二级联动
2019/11/27 Javascript
vue实现按钮切换图片
2021/01/20 Vue.js
用tensorflow实现弹性网络回归算法
2018/01/09 Python
python实现excel公式格式化的示例代码
2020/12/23 Python
详解html5页面 rem 布局适配方法
2018/01/12 HTML / CSS
捷克领先的户外服装及配件市场零售商:ALPINE PRO
2018/01/09 全球购物
Laura官网:加拿大女性的顶级时尚目的地
2019/09/20 全球购物
荷兰美妆护肤品海淘网站:Beautinow(中文)
2020/11/22 全球购物
参观监狱心得体会
2014/01/02 职场文书
商务考察邀请函范文
2014/01/21 职场文书
上课看小说检讨书
2014/02/22 职场文书
应届硕士毕业生自荐信
2014/05/26 职场文书
信息工作经验交流材料
2014/05/28 职场文书
激励员工的口号
2014/06/16 职场文书
大学毕业生个人自荐书
2014/07/02 职场文书
党员一帮一活动总结
2014/07/08 职场文书
委托书的写法
2014/08/30 职场文书
MySQL七大JOIN的具体使用
2022/02/28 MySQL