学习 NodeJS 第八天:Socket 通讯实例


Posted in NodeJs onDecember 21, 2016

前言

一般来讲,HTTP 是基于文本的“单向”通讯机制。这里所谓的“单向”,乃相对于“双向”而言,因为 HTTP 服务器只需根据请求返还恰当的 HTML 给客户端即可,不涉及客户端向服务端的通讯。这种单向的机制比较简单,对网络质量要求也不高。而更多的场景则是需要可靠、稳定的端到端连接。一般这种服务是实时的、有态的而且是长连接,长连接则暗示两段须达致相向通讯的能力,也就说是服务端客户端两者间能够实时地相互间通信。毫无疑问,能够实时通信的服务器正是我们对服务器基本要求之一。区别于 HTTP 服务器以 HTTP 为通讯协议, 实时服务器一般采用较为底层的 TCP/IP 为协议通讯,实现了“套字节 Socket”的双向机制。

Socket 是根据博克莱 (U.C.Berkley) 大学早期发展的 Socket 概念写成的,其设计理念是是将网络传输类比成文件的读取与写入 (传送的动作被视为是写入/接收的动作被视为是读取),如此、传送与接收就简化为编程人员比较容易懂的 读取与写入,降低了网络编程的学习困难度。

聊天室服务器

聊天室的实时连接基于底层的 TCP 直接连接,为此我们须调用 Node 的 TCP 模块。如果不太熟悉所谓 TCP 网络编程?太底层了是不是?没关系,我也不熟悉,边学边做嘛,只不过千万不必因为遇到陌生的词汇而害怕,其实这样原理并不深奥,而且下面的例子也十分的简单易懂!咱们就从最简单的开始吧,下面代码仅仅十行,它的作用是服务器向客户端输出一段文本,完成 Sever --> Client 的单向通讯。

// Sever --> Client 的单向通讯 
var net = require('net'); 
 
var chatServer = net.createServer(); 
 
chatServer.on('connection', function(client) { 
 client.write('Hi!\n'); // 服务端向客户端输出信息,使用 write() 方法 
 client.write('Bye!\n'); 
 
 client.end(); // 服务端结束该次会话 
}); 
 
chatServer.listen(9000);

客户端可以是系统自带的 Telnet:

telnet 127.0.0.1 9000

执行 telnet 后,与服务点连接,反馈 Hi! Bye! 的字符,并立刻结束服务端程序终止连接。如果我们要服务端接到到客户端的信息?可以监听 server.data 事件并且不要中止连接(否则会立刻结束无法接受来自客户端的消息):

// 在前者的基础上,实现 Client --> Sever 的通讯,如此一来便是双向通讯 
var net = require('net'); 
var chatServer = net.createServer(),  
 clientList = []; 
  
chatServer.on('connection', function(client) { 
 // JS 可以为对象自由添加属性。这里我们添加一个 name 的自定义属性,用于表示哪个客户端(客户端的地址+端口为依据) 
 client.name = client.remoteAddress + ':' + client.remotePort; 
 client.write('Hi ' + client.name + '!\n'); 
 clientList.push(client); 
 client.on('data', function(data) {  
  broadcast(data, client);// 接受来自客户端的信息 
 }); 
}); 
function broadcast(message, client) { 
 for(var i=0;i<clientList.length;i+=1) {  
  if(client !== clientList[i]) {  
  clientList[i].write(client.name + " says " + message);  
  } 
 } 
} 
chatServer.listen(9000);

这里要说明一下的是,不不同操作系统对端口范围的限制不一样,有可能是随机的。

那么上面是不是一个完整功能的代码呢?我们说还有一个问题没有考虑进去:那就是一旦某个客户端退出,却仍保留在 clientList 里面,这明显是一个空指针(NullPoint)。如果是在这样的话我们写程序太脆弱了,能不能更健壮一些?——请接着看。

首先我们简单地把 client 从数组 clientList 中移除掉。完成这工作一点都不困难。Node TCP API 已经为我们提供了 end 事件,即客户端中止与服务端连接的时候发生。移除 client 对象的代码如下:

chatServer.on('connection', function(client) { 
 client.name = client.remoteAddress + ':' + client.remotePort 
 client.write('Hi ' + client.name + '!\n'); 
 
 clientList.push(client) 
 
 client.on('data', function(data) { 
 broadcast(data, client) 
 }) 
 
 client.on('end', function() { 
 clientList.splice(clientList.indexOf(client), 1); // 删除数组中的制定元素。这是 JS 基本功哦~ 
 }) 
})

但是我们还不敢说上述代码很健壮,因为一旦 end 没有被触发,异常仍然存在着。下面我们看看解决之道:重写 broadcast():

function broadcast(message, client) { 
 var cleanup = [] 
 for(var i=0;i<clientList.length;i+=1) { 
 if(client !== clientList[i]) { 
 
  if(clientList[i].writable) { // 先检查 sockets 是否可写 
  clientList[i].write(client.name + " says " + message) 
  } else { 
  cleanup.push(clientList[i]) // 如果不可写,收集起来销毁。销毁之前要 Socket.destroy() 用 API 的方法销毁。 
  clientList[i].destroy() 
  } 
 
 } 
 } //Remove dead Nodes out of write loop to avoid trashing loop index 
 for(i=0;i<cleanup.length;i+=1) { 
 clientList.splice(clientList.indexOf(cleanup[i]), 1) 
 } 
}

TCP API 中还提供一个 error 事件,用于捕捉客户端的异常:

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

Node 网络编程的 API 还丰富,此次仅仅是个入门,更多的内容请接着看,关于浏览器 Socket 应用。

Socket.IO

前面说到,浏览器虽然也属于客户端的一种,但仅支持“单工”的 HTTP 通讯。有见及此,HTML5 新规范中推出了基于浏览器的 WebSocket,开发了底层的接口,允许我们能进行 更强大的操作,超越以往的 XHR。

如第一个例子那般,我们无须第三方框架就可以直接与 Node TCP 服务器 进行 Socket  通讯。

但我们又要认清一个事实,不是每个浏览器都可以顺利支持 WebSocket 的。于是 Socket.IO (http://socket.io)出现了,它提供了不支持 WebSocket 时候的降级支持,同时使得一些旧版本的浏览器也可以“全双工”地工作。优先使用的顺序如下:

  • WebSocket
  • Socket over Flash API
  • XHR Polling 长连接
  • XHR Multipart Streaming
  • Forever Iframe
  • JSONP Polling

经过封装,我们可以不探究客户端使用上述哪一种技术达致“全双工”;而我们编写代码时,亦无论考虑哪种放法,因为 Socket.IO 给我们的 API 只有一套。了解 Socket.IO 其用法就可以了。

先在浏览器部署 Socket.IO 的前端代码:

<!DOCTYPE html> 
<html> 
 <body> 
 <script src="/socket.io/socket.io.js"></script> 
 <script> 
  var socket = io.connect('http://localhost:8080'); 
  // 当服务端发送一条消息到客户端,message 事件即被触发。我们把消息在控制台打印出来 
  socket.on('message', function(data){ console.log(data) }) 
 </script> 
 </body> 
</html>

服务端 Node 代码:

var http = require('http'), 
io = require('socket.io'), 
fs = require('fs'); 
 
// 虽然我们这里使用了同步的方法,那会阻塞 Node 的事件循环,但是这是合理的,因为 readFileSync() 在程序周期中只执行一次,而且更重要的是,同步方法能够避免异步方法所带来的“与 SocketIO 之间额外同步的问题”。当 HTML 文件读取完毕,而且服务器准备好之后,如此按照顺序去执行就能让客户端马上得到 HTML 内容。 
var sockFile = fs.readFileSync('socket.html'); 
 
// Socket 服务器还是构建于 HTTP 服务器之上,因此先调用 http.createServer() 
server = http.createServer(); 
server.on('request', function(req, res){ 
 // 一般 HTTP 输出的格式 
 res.writeHead(200, {'content-type': 'text/html'}); 
 res.end(sockFile); 
}); 
 
server.listen(8080); 
 
var socket = io.listen(server); // 交由 Socket.io 接管 
 
// Socket.io 真正的连接事件 
socket.on('connection', function(client){ 
 console.log('Client connected'); 
 client.send('Welcome client ' + client.sessionId); // 向客户端发送文本 
});

当客户端连接时,服务端会同时出发两个事件:server.onRequest 和 Socket.onConnection。它们之间有什么区别呢?区别在于 Socket 的是持久性的。

多个 Socket 连接,先是客户端代码:

<!DOCTYPE html> 
<html> 
 <body> 
 <script src="/socket.io/socket.io.js"></script> 
 <script> 
  var upandrunning = io.connect('http://localhost:8080/upandrunning'); 
  var weather = io.connect('http://localhost:8080/weather'); 
  upandrunning.on('message', function(data){ 
  document.write('<br /><br />Node: Up and Running Update<br />'); 
  document.write(data); 
  }); 
  weather.on('message', function(data){ 
  document.write('<br /><br />Weather Update<br />'); 
  document.write(data); 
  }); 
 </script> 
 </body> 
</html>

服务端代码:

var sockFile = fs.readFileSync('socket.html'); 
 
server = http.createServer(); 
server.on('request', function(req, res){ 
 res.writeHead(200, {'content-type': 'text/html'}); 
 res.end(sockFile); 
}); 
 
server.listen(8080); 
 
var socket = io.listen(server); 
 
socket.of('/upandrunning') 
 .on('connection', function(client){ 
 console.log('Client connected to Up and Running namespace.'); 
 client.send("Welcome to 'Up and Running'"); 
}); 
 
socket.of('/weather') 
 .on('connection', function(client){ 
 console.log('Client connected to Weather namespace.'); 
 client.send("Welcome to 'Weather Updates'"); 
});

 如上代码,我们可以划分多个命名空间,分别是 upandrunning 和 weather。

关于 Express 中使用 Soclet.io,可以参考《Node:Up and Ruuning》一书的 7.2.2 小节。

今晚时间的关系,涉及 Socket.io 许多方面还没有谈,容小弟我日后再了解。

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

NodeJs 相关文章推荐
nodejs教程之环境安装及运行
Nov 21 NodeJs
浅析nodejs实现Websocket的数据接收与发送
Nov 19 NodeJs
使用nodejs中httpProxy代理时候出现404异常的解决方法
Aug 15 NodeJs
Express与NodeJs创建服务器的两种方法
Feb 06 NodeJs
搭建简单的nodejs http服务器详解
Mar 09 NodeJs
nodejs入门教程一:概念与用法简介
Apr 24 NodeJs
nodejs中sleep功能实现暂停几秒的方法
Jul 12 NodeJs
Nodejs+express+ejs简单使用实例代码
Sep 18 NodeJs
nodeJs爬虫的技术点总结
May 13 NodeJs
基于nodejs res.end和res.send的区别
May 14 NodeJs
nodejs对项目下所有空文件夹创建gitkeep的方法
Aug 02 NodeJs
Nodejs环境实现socket通信过程解析
Jul 03 NodeJs
详解Nodejs基于mongoose模块的增删改查的操作
Dec 21 #NodeJs
nodejs redis 发布订阅机制封装实现方法及实例代码
Dec 15 #NodeJs
解析NodeJs的调试方法
Dec 11 #NodeJs
nodejs连接mongodb数据库实现增删改查
Dec 01 #NodeJs
Nodejs 搭建简单的Web服务器详解及实例
Nov 30 #NodeJs
Nodejs下用submit提交表单提示cannot post错误的解决方法
Nov 21 #NodeJs
Nodejs进阶:核心模块net入门学习与实例讲解
Nov 21 #NodeJs
You might like
php中的MVC模式运用技巧
2007/05/03 PHP
php设计模式 Proxy (代理模式)
2011/06/26 PHP
codeigniter自带数据库类使用方法说明
2014/03/25 PHP
php连接odbc数据源并保存与查询数据的方法
2014/12/24 PHP
php模拟登陆的实现方法分析
2015/01/09 PHP
ThinkPHP实现的rsa非对称加密类示例
2018/05/29 PHP
jQuery Tips 为AJAX回调函数传递额外参数的方法
2010/12/28 Javascript
利用了jquery的ajax实现二级联互动菜单
2013/12/02 Javascript
Get中文乱码IE浏览器Get中文乱码解决方案
2013/12/26 Javascript
javascript实现数独解法
2015/03/14 Javascript
javascript检查某个元素在数组中的索引值
2016/03/30 Javascript
SWFUpload多文件上传及文件个数限制的方法
2016/05/31 Javascript
Vue form 表单提交+ajax异步请求+分页效果
2017/04/22 Javascript
微信小程序canvas写字板效果及实例
2017/06/15 Javascript
微信、QQ、微博、Safari中使用js唤起App
2018/01/24 Javascript
详解vue通过NGINX部署在子目录或者二级目录实践
2018/09/03 Javascript
解决axios会发送两次请求,有个OPTIONS请求的问题
2018/10/25 Javascript
小程序实现单选多选功能
2018/11/04 Javascript
详解vue中$nextTick和$forceUpdate的用法
2019/12/11 Javascript
Vue实现星级评价效果实例详解
2019/12/30 Javascript
解决echarts图表使用v-show控制图表显示不全的问题
2020/07/19 Javascript
[42:32]DOTA2上海特级锦标赛B组资格赛#2 Fnatic VS Spirit第二局
2016/02/27 DOTA
[56:35]DOTA2上海特级锦标赛主赛事日 - 5 总决赛Liquid VS Secret第一局
2016/03/06 DOTA
python开发简易版在线音乐播放器
2017/03/03 Python
python-opencv 将连续图片写成视频格式的方法
2019/01/08 Python
Python列表元素常见操作简单示例
2019/10/25 Python
图解python全局变量与局部变量相关知识
2019/11/02 Python
HTML5给汉字加拼音收起展开组件的实现代码
2020/04/08 HTML / CSS
Trunki英国官网:儿童坐骑式行李箱
2017/05/30 全球购物
Parfumdreams芬兰:购买香水和化妆品
2021/02/13 全球购物
高中生学习的自我评价
2013/12/14 职场文书
期中考试反思800字
2014/05/01 职场文书
庆六一文艺汇演活动方案
2014/08/26 职场文书
电影复兴之路观后感
2015/06/02 职场文书
Python关于OS文件目录处理的实例分享
2021/05/23 Python
python中redis包操作数据库的教程
2022/04/19 Python