webSocket协议(聊天室)

1)HTTP无法轻松实现实时应用:

1、 HTTP协议是无状态的,服务器只会响应来自客户端的请求,但是它与客户端之间不具备持续连接。


2、我们可以非常轻松的捕获浏览器上发生的事件(比如用户点击了盒子),这个事件可以轻松产生与服务器的数据交互(比如Ajax)。但是,反过来却是不可能的:服务器端发生了一个事件,服务器无法将这个事件的信息实时主动通知它的客户端。只有在客户端查询服务器的当前状态的时候,所发生事件的信息才会从服务器传递到客户端。

2)以往解决实时应用方法

1、 长轮询:客户端每隔很短的时间,都会对服务器发出请求,查看是否有新的消息,只要轮询速度足够快,例如1秒,就能给人造成交互是实时进行的印象。这种做法是无奈之举,实际上对服务器、客户端双方都造成了大量的性能浪费。


2、长连接:客户端只请求一次,但是服务器会将连接保持,不会返回结果(想象一下我们没有写res.end()时,浏览器一直转小菊花)。服务器有了新数据,就将数据发回来,又有了新数据,就将数据发回来,而一直保持挂起状态。这种做法的也造成了大量的性能浪费。

3)WebSocket协议能够让浏览器和服务器全双工实时通信,互相的,服务器也能主动通知客户端了。

WebSocket的原理非常的简单:利用HTTP请求产生握手,HTTP头部中含有WebSocket协议的请求,所以握手之后,二者转用TCP协议进行交流(QQ的协议)。现在的浏览器和服务器之间,就是QQ和QQ服务器的关系了。


所以WebSocket协议,需要浏览器支持,更需要服务器支持。
● 支持WebSocket协议的浏览器有:Chrome 4、火狐4、IE10、Safari5
● 支持WebSocket协议的服务器有:Node 0、Apach7.0.2、Nginx1.3

4)vue3.x+typescript+node实现socket通信

前端:

<template>
  <div>
    <div>
      <input type="text" v-model="msg" />
      <button type="button" @click="handelMsg">提交</button>
    </div>
    <div class="msg_lis">
      <div
        class="item"
        :class="{ active_user: userName === item.userName }"
        v-for="item of list"
        :key="item.id"
      >
        <dl>
          <dt>{{ item.createTime }}</dt>
          <dd>
            <span>{{ item.userName }}:</span>{{ item.msg }}
          </dd>
        </dl>
      </div>
    </div>
  </div>
</template>
<script lang="ts">
interface IListItem {
  id: number;
  createTime: string;
  msg: string;
  userName: string;
}
interface IData {
  msg: string;
  list: IListItem[];
  userName: string;
}
import { reactive, toRefs } from "vue";
import hookSocket from "../hook/hook_socket";
export default {
  setup() {
    const data: IData = reactive({
      list: [],
      msg: "",
      userName: localStorage.getItem("userName") || "",
    });
    const ws = hookSocket((a: MessageEvent) => {
      if (!a.data) return;
      data.list.push(<IListItem>JSON.parse(a.data));
    });
    const handelMsg = () => {
      if (data.msg == "") {
        alert("请输入消息");
        return;
      }
      ws.send(
        JSON.stringify(<IListItem>{
          id: new Date().getTime(),
          createTime: new Date().toLocaleString().replace(/\//g, "-"),
          msg: data.msg,
          userName: data.userName,
        })
      );
      data.msg = "";
    };
    return {
      ...toRefs(data),
      handelMsg,
    };
  },
};
</script>

<style>
.active_user {
  color: red;
}
</style>
import { WS_URL } from '../config/ws_config'
//const WS_URL = `ws://localhost:8000`;
export default function hookSocket(handleMsg: (ev: MessageEvent) => void) {
    const WS = new WebSocket(WS_URL);//浏览器自带WebSocket对象
    const init = () => {
        bindEvent();
    }
    init();
    function bindEvent() {
        WS.addEventListener('open', function (event) {
            console.log(`WebSocket Open`);
        }, false)
        WS.addEventListener('message', handleMsg, false)
        WS.addEventListener('close', function () { console.log(`WebSocket close`); }, false)
        WS.addEventListener('error', function () { console.error(`WebSocket error`); }, false)
    }
    return WS;
}

后端:

使用nodeman开启服务,并安装ws模块

const WebSocket = require('ws');
; ((ws) => {
    const server = new ws.Server({ port: 8000 });
    const init = () => {
        bindEvent();
    }
    init();
    function bindEvent() {
        server.on('open', function () {
            console.log(`WebSocket open`);
        })
        server.on('close', function () { console.log(`WebSocket close`); })
        server.on('error', function () { console.log(`WebSocket error`); })
        server.on('connection', handleConnection)
    }
    function handleConnection(ws) {
        console.log('WebSocket Connection');
        ws.on('message', handleMessage)
    }
    function handleMessage(msg){
        //获取每个链接的客户端 把消息广播出去。
        server.clients.forEach(c=>{
            c.send(msg);
        })
    }
})(WebSocket)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,539评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,594评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,871评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,963评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,984评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,763评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,468评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,357评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,850评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,002评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,144评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,823评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,483评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,026评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,150评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,415评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,092评论 2 355

推荐阅读更多精彩内容