socket.io的使用

简单介绍一下如何在nodejs上使用socket.io,以下栗子来自官网,细节上有点小修改。

scoket基本知识

首先介绍一下socket的基本知识:网络上的两个程序通过一个双向的通信实现数据的交换,这个连接的一端称为一个socket(端口号),socket的本质是编程接口(API),对TCP/IP的封装TCP/IP提供程序员做网络开发时可用的接口。HTTP是轿车,提供了封装或者显示数据的具体形式;Socket是发动机,提供了网络通信的能力。

socket用于描述ip地址和端口。不同的端口对应不同的服务,每个socket都会绑定一个端口。就像插座有110伏也有220伏,插头(socket)插入(绑定)不同的编号的插座(端口),就可以得到不同的服务

socket.io

socket.io是一个实现实时web通信的JavaScript库。它包含两部分,在浏览器上运行的客户端库和在nodejs上运行的服务器端库。

ps:以下用到的fn均是Function

安装socket.io

$ npm install socket.io

在Node http server上使用,官方小栗子

服务器端
var http = require('http');
var fs = require('fs');
function onRequest(req, res) {
  fs.readFile(__dirname + '/index.html', function(err, data) {
    if (err) {
      res.writeHead(500);
       return res.end('error loading index.html');
    }
    res.writeHead(200);
    res.end(data);
  })
}

var app = http.createServer(onRequest);
var io = require('socket.io')(app);
io.on('connection', function(socket) {
  socket.emit('news', 'Hello world');
  socket.on('my other event', function(data) {
    console.log(data);
  })
})

app.listen(8001, function() {
  console.log('listen on 8001')
})
客户端
<head>
  <title>using with node http server</title>
  <script src="/socket.io/socket.io.js"></script>
  <script src="http://code.jquery.com/jquery-1.11.1.js"></script>
</head>
<body>
  <p id="msg">this is a block</p>
  <script>
    var socket = io();
    socket.on('news', function(data) {
      console.log(data);
      $('#msg').text(data)
      socket.emit('my other event', {my: 'data'})
    })
  </script>
</body>

在express框架中使用,官方小栗子

服务器端
var app = require('express')();
var http = require('http').Server(app);
var io = require('socket.io')(http);

app.get('/', function(req, res) {
  res.sendFile(__dirname + '/index.html')
})

io.on('connection', function(socket) {
  socket.emit('news', 'Hello world');
  socket.on('my other event', function(data) {
    console.log(data);
  })
})

http.listen(8001, function() {
  console.log('listen on 8001')
})
客户端

同nodeJs的客户端。

Server API

创建server

通过·require('socket.io')暴露出Server, Server创建的几种方式有以下三种,其中第一参数和第二参数都是可选项

  1. Server(): require('socket.io')();
  2. Server(opts:Object): 通过opts创建,opts包含的选项有:
    serverClient、path以及创建engine.io Server的options
  3. Server(server, opts:Object): 通过已有的http server创建新的Server
  4. Server(port, opts:Object): 通过监听的端口号创建

server的方法和属性

ps: 未注明返回值的,返回值为Server
serveClient(v:Boolean): 如果v的值是true,服务器将会对客户端文件提供服务, socket.io的客户端文件会压缩至vendor.js中,所以一般生产环境里,serveClient设置成false。

var io = require('socket.io')(httpServer, {
 serveClient: (config.env === 'production') ? false : true 
});

path(v:String): 静态文件的路径,默认/socket.io

adapter(v:Adapter):设置 rooms的适配器,默认socket.io-adapter

origin(v:String):设置默认的origins,默认所有的origins都允许

attach(server, opts:Object): 通过提供的opts,将server与提供的http server合并,返回一个新的server

attach(port:Number, opts:Object):将port和opts附着至Server上,返回新的server

sockets:返回值Namespace, 默认/

listen: 与创建该server的http server的listen方法一致

bind(srv: engine#Server): 将server绑定值一个指定的engine server上,返回新的Server

onconnection(socket: engine#socket)

of(nsp:String): 返回Namespace

use: 同Namespace的use方法

emit: 对所有连接的客户端发送一个事件

var io = require('socket.io')();
io.sockets.emit('an event sent to all connected clients');
// 等同于
io.emit('an event sent to all connected clients');

of(namespace):Namespace: 通过路径标识的nsp,初始化Namespace

Namespace

socket.io允许定义socket的namespace(命名空间),namespace意味着给socket分配不同的路径,如var socket = io('/example1');默认情况下是'/'

index.js

var ep1 = io.of('/example1');
ep1.on('connection', function(socket) {
  console.log('example1 connection')
  socket.emit('news', 'example1 connection');
})

io.on('connection', function(socket) {
  console.log('io connection')
  socket.emit('news', 'io connection');
})

index.html

 var socket = io();
 socket.on('news', function(data) {
   console.log(data);
   $('#msg').text(data)
   socket.emit('my other event', {my: 'data'})
 })

example1.html

 var socket = io('/example1');
 socket.on('news', function(data) {
   console.log(data);
   $('#msg').text(data)
   socket.emit('my other event', {my: 'data'})
 })

观察结果,
进入example1页面,会打印出example1 connection,
进入任何页面,都会打印出io connection

事件

connection/connect:连接时触发
socket:客户端返回的socket

属性

name:String:命名空间标识属性
connected:连接至这个命名空间的socket的hash值,通过id进行索引
use(fn):注册中间件fn,客户端返回socket时执行,socket和next作为这个中间件fn的参数,next的参数也是fn。

var io = require('socket.io')();
io.use(function(socket, next){
  if (socket.request.headers.cookie) return next();
  next(new Error('Authentication error'));
});

Socket

socket是通信的基础类,一个socket属于一个特定的Namespace,并且在隐藏的Client下进行通信。

rooms:Array: socket所在的rooms列表。

client:Client: 隐藏的Client。

conn:Socket: Client下正在传送连接的socket对象

request: 返回Request对象

id: socket会话的唯一标识

emit: 向特定socket发送事件

var io = require('socket.io')();
io.on('connection', function(socket){
  socket.emit('an event', { some: 'data' });
});

broadcast:向除了正在连接的socket以外的其他已经连接的socket发送事件

服务器端:

io.on('connection', function(socket) {
  client++;
  socket.broadcast.emit('newClientConnect', client + ' clients connects')
  socket.emit('newClientConnect', 'Hey, welcome');
  socket.on('disconnect', function() {
    client--;
  })
})

客户端:

 var socket = io();
 socket.on('newClientConnect', function(data) {
   console.log(data);
 })

join(name: String, fn):Socket:socket加入room时候触发,fn是callback

leave(name: String, fn:Socket:socket离开room时候触发fn是callback

to/in(room:String):Socket:为后续的emit事件做准备,只限定于加入该room的sockets

var io = require('socket.io')();
in.on('connection', function(socket) {
  socket.to('others').emit('an event', {some: 'data'});
});

Client API

Manager

一个manager代表一个给定socket.io服务器的一次连接,除非multiplex设置为false(即:强制创建新的连接),在一次连接中,可能会有多个socket,所以一个manager可能对应多个socket。

参考文献:
https://www.tutorialspoint.com/socket.io/index.htm

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容