WebSocket


websocket

这篇总结主要介绍一下websocket

常见的消息推送方案

1.轮询
2.长轮询
3.SSE
4.websocket

轮询方式

轮询:指的是浏览器以指定的时间间隔向服务器发出http请求,服务器实时返回数据给浏览器
长轮询:浏览器发出ajax请求,服务器端接收请求后,如果没有数据,则会阻塞请求直到有数据或者超时才会返回

SSE(server-sent event):服务器发送事件

1.SSE在服务器和客户端之间打开了一个单项通道,是服务器给浏览器的
2.服务器响应的不再是一次性的数据包,即不是把数据一次性返回回来,而是text/event-stream类型的数据流信息
3.服务器有数据变更时将以数据流式传输到客户端(数据流式指的就是有源源不断的数据从服务器响应给客户端)

websocket

1.websocket介绍

是什么?websocket是一种基于TCP连接上进行全双工协议
什么是全双工协议?:允许数据在两个方向上同时传输
什么是半双工协议?:允许数据在两个方向上传输,但是同一个时间段内只允许一个方向上传输

2.websocket原理

浏览器给服务器发送http的协议请求,这个请求头中带有UPgrade.websocket,要求将协议升级为websocket协议。
服务器响应头中的响应状态码为101 Switching Protocls,此时就将http协议转换为websocket协议了。
再让浏览器发送下一次请求时开始,都是发送的websocket协议的请求了

3.浏览器websocket API (即前端应该怎么做?)

1.websocket 相关事件

open事件 ws.onopen 连接建立时触发
message事件 ws.onmessage 浏览器接收到服务器发送的数据时触发
close事件 ws.onclose 连接关闭时触发
error事件 ws.onerror 在WebSocket连接发生错误时触发。

2.websocket 对象提供的方法

send(数据) 通过websocket对象调用该方法发送数据给服务器

3.步骤:

1.创建websocket对象

let ws = new WebSocket(URL)
ws.onopen = function(){
    //连接建立时给个消息提示
}
document.querySelector('#btn1').onclick = function () {
     //拿到点击后输入框数据
     let msg = document.querySelector('input').value
     // 发送消息,调用ws对象的send方法,发送输入框数据
     ws.send(msg)
     //渲染页面
     content.innerHTML += `用户:${msg} <br>`
   }
ws.onmessage = function(evt){
    //通过evt.data可以获取服务器发送的数据。同时通过evt.data渲染数据
}
ws.onclose = function(){
    //连接关闭时给个消息提示
}

URL这个参数该如何去写?
格式:协议://ip地址/访问路径
协议:协议名应为ws

4.原生js实现机器人客服

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Document</title>
  </head>
  <body>
    <div class="box">
      <input type="text" placeholder="请输入聊天内容" />
      <button id="btn1">发送消息</button>
      <button id="btn2">退出聊天室</button>
      <div class="content"></div>
    </div>

    <script>
      let content = document.querySelector('.content')

      // 1.与服务器进行WebSocket链接
      // wss://echo.websocket.org : 这个接口你给它发什么,它就回复你什么
      const ws = new WebSocket('ws://echo.websocket.org') // 建立与服务端地址的连接

      // 2.连接成功回调
      ws.onopen = function () {
        // 如果执行此函数 表示与服务器建立关系成功
        content.innerHTML += '提示:<p style="color:red">websockt链接成功</p>'
      }

      //3.发送消息
      document.querySelector('#btn1').onclick = function () {
        console.log(1111)

        let msg = document.querySelector('input').value
        // 发送消息
        ws.send(msg)
        //渲染页面
        content.innerHTML += `用户:${msg} <br>`
      }

      //4.接收消息
      ws.onmessage = function (event) {
        // event中的data就是服务器发过来的消息
        console.log(event);
        
        content.innerHTML += `服务器:${event.data} <br>`
      }

      //5.关闭连接
      document.querySelector('#btn2').onclick = function () {
        ws.close()
      }

      //6.成功关闭链接回调
      ws.onclose = function () {
        // 关闭连接成功
        content.innerHTML += '提示:<p style="color:red">websockt关闭成功</p>'
      }
    </script>
  </body>
</html>

5.vue中如何使用?

下载:使用vue ui安装,或npm i socket.io-client

  • 1.连接服务器

  • 2.连接成功回调

    • socket.on('connect', () => {
         console.log('建立连接成功了')
      })
      
      
      - 3.接收服务器消息
      
        - ```javascript
          socket.on('message', msg => {})
  • 4.给服务器发消息

    • socket.emit('message', {
        // 消息
        msg: '消息内容',
        // 时间戳
        timestamp: Date.now()
      })
      
      
      - 5.断开链接
      
        - ```javascript
          socket.disconnect()
<template>
  <div class="student-container">
    <van-nav-bar
      fixed
      left-arrow
      @click-left="$router.back()"
      title="机器人"
    ></van-nav-bar>
    <div class="chat-list">
      <template v-for="item in message">
        <!-- 左侧是机器人 -->
        <div
          class="chat-item left"
          :key="item.timestamp"
          v-if="item.isStu === true"
        >
          <van-image fit="cover" round :src="littleQ" />
          <div class="chat-pao">{{ item.msg }}</div>
        </div>

        <!-- 右侧是当前用户 -->
        <div class="chat-item right" :key="item.timestamp" v-else>
          <div class="chat-pao my">{{ item.msg }}</div>
          <van-image fit="cover" round :src="self" />
        </div>
      </template>
    </div>
    <div class="reply-container van-hairline--top">
      <van-search v-model.trim="word" placeholder="说点什么..." @search="send">
      </van-search>
    </div>
  </div>
</template>

<script>
// 导入精美的图片
import littleQ from '@/assets/student.jpg'
import self from '@/assets/teacher.png'
// socket.io的客户端
import io from 'socket.io-client'
export default {
  name: 'student',
  data () {
    return {
      word: '',
      message: [], // 聊天信息
      littleQ, // 机器人
      self, // 自己
      socket: undefined // socket对象
    }
  },
  created () {
    // 1.开始连接
    this.socket = io('http://118.24.62.36:8888')
    // 2.链接成功回调
    this.socket.on('connect', () => {
      console.log('建立连接成功了')
    })

    // 3.接收服务器返回数据
    this.socket.on('message', msg => {
      console.log('接收到消息了')
      console.log('msg:', msg)
      this.message.push({
        ...msg,
        // 是机器人
        isStu: true
      })
    })
  },
  methods: {
    //4.发送消息
    send () {
      console.log(this.word)
      if (this.word !== '') {
        this.socket.emit('message', {
          // 消息
          msg: this.word,
          // 时间戳
          timestamp: Date.now()
        })
        // 本地数组也要更新
        this.message.push({
          msg: this.word,
          // 时间戳
          timestamp: Date.now(),
          // 不是机器人
          isStu: false
        })
        this.word = ''
      }
    }
  }
}
</script>

<style lang="less">
.student-container {
  height: 100%;
  width: 100%;
  position: absolute;
  left: 0;
  top: 0;
  box-sizing: border-box;
  background: #fafafa;
  padding: 46px 0 50px 0;
  .van-nav-bar {
    background-color: #3296fa;
    .van-nav-bar__title {
      color: white;
    }
    .van-icon {
      color: white;
    }
  }
  .chat-list {
    height: 100%;
    overflow-y: scroll;
    .chat-item {
      padding: 10px;
      .van-image {
        vertical-align: top;
        width: 40px;
        height: 40px;
      }
      .chat-pao {
        vertical-align: top;
        display: inline-block;
        min-width: 40px;
        max-width: 70%;
        min-height: 40px;
        line-height: 38px;
        border: 0.5px solid #c2d9ea;
        border-radius: 4px;
        position: relative;
        padding: 0 10px;
        background-color: #e0effb;
        word-break: break-all;
        font-size: 14px;
        color: #333;
        &::before {
          content: '';
          width: 10px;
          height: 10px;
          position: absolute;
          top: 12px;
          border-top: 0.5px solid #c2d9ea;
          border-right: 0.5px solid #c2d9ea;
          background: #e0effb;
        }
      }
      .chat-pao.my {
        background-color: #9eea6a;
        &::before {
          content: '';
          background: #9eea6a;
        }
      }
    }
  }
  .chat-item.right {
    text-align: right;
    .chat-pao {
      margin-left: 0;
      margin-right: 15px;
      &::before {
        right: -6px;
        transform: rotate(45deg);
      }
    }
  }
  .chat-item.left {
    text-align: left;
    .chat-pao {
      margin-left: 15px;
      margin-right: 0;
      &::before {
        left: -5px;
        transform: rotate(-135deg);
      }
    }
  }
  .reply-container {
    position: fixed;
    left: 0;
    bottom: 0;
    height: 44px;
    width: 100%;
    background: #f5f5f5;
    z-index: 9999;
  }
}
</style>

文章作者: 冷杨威
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 冷杨威 !
  目录
-->