Websocket获取B站直播间弹幕教程——第二篇、解包/拆包

news2025/1/21 0:46:15

教程一、Websocket获取B站直播间弹幕教程 — 哔哩哔哩直播开放平台

1、封包

我们连接上B站Websocket成功后,要做两件事情:

  • 第一、发送鉴权包。
  • 第二、发送心跳包,每30秒一次,维持websocket连接。

这两个包不是直接发送过去,而是要创建byte数组,将一些数据 按B站协议格式 用大端序写入到byte数组

协议

在这里插入图片描述

  • 1、(4 byte)Packet Length:整个Packet的长度,包含Header。
  • 2、(2 byte)Header Length:Header的长度,固定为16。
  • 3、(2 byte)Version
    • 如果Version=0,Body中就是实际发送的数据。
    • 如果Version=2,Body中是经过压缩后的数据,请使用zlib解压,然后按照Proto协议去解析。
  • 4、(4 byte)Operation:消息的类型:
    • Operation == 2,客户端发送的心跳包(30秒发送一次)
    • Operation == 3 ,服务器收到心跳包的回复
    • Operation == 5 ,服务器推送的弹幕消息包
    • Operation == 7 ,客户端发送的鉴权包(客户端发送的第一个包)
    • Operation == 8 ,服务器收到鉴权包后的回复
  • 5、(4 byte)Sequence ID:保留字段,可以忽略。
  • 6、(? byte)Body:消息体
    • body为json格式字符串 --> 转Byte数组
    • 如果是心跳包body就为空

例子

这边以JAVA代码为例。
代码依赖了FastJson2,用于将Json(Map)转Json字符串。

public static byte[] pack(String jsonStr, short code){
    byte[] contentBytes = new byte[0];
    //如果是鉴权包,那一定带有jsonStr
    if(7 == code){
        contentBytes = jsonStr.getBytes();
    }
    try(ByteArrayOutputStream data = new ByteArrayOutputStream();
        DataOutputStream stream = new DataOutputStream(data)){
        stream.writeInt(contentBytes.length + 16);//封包总大小
        stream.writeShort(16);//头部长度 header的长度,固定为16
        stream.writeShort(0);//Version, 客户端一般发送的是普通数据。
        stream.writeInt(code);//操作码(封包类型)
        stream.writeInt(1);//保留字段,可以忽略。
        if(7 == code){
            stream.writeBytes(jsonStr);
        }
        return data.toByteArray();
    }
}

这样封包的方法就写好了,jsonStr为要发的数据,code为包的类型。

定义生成鉴权包的方法:
public byte[] generateAuthPack(String jsonStr) throws IOException {
   return pack(jsonStr, 7);
}

如果你是非官方开放API接口调用,那jsonStr得自己生成。

public byte[] generateAuthPack(String uid, String buvid,String token, int roomid){
   JSONObject jo = new JSONObject();
    jo.put("uid", uid);
    jo.put("buvid", buvid);
    jo.put("roomid", roomid);
    jo.put("protover", 0);
    jo.put("platform", "web");
    jo.put("type", 2);
    jo.put("key", token);
    return pack(jo.toString(), 7);
}
  • 参数
    • uid : 你Cookie的DedeUserID
    • buvid : 你Cookie的buvid3
    • token : 鉴于是否登录token
    • roomid : 直播间ID

如何获取Token?
----> 【JAVA版本】最新websocket获取B站直播弹幕——非官方API

定义生成心跳包的方法:
public static byte[] generateHeartBeatPack() throws IOException {
    return pack(null, 2);
}

2、解包

成功鉴权后,我们获取到B站数据也都是byte数组,格式跟上面一样,按上面的格式来读取就行了。
不过如果是Zip包那稍微麻烦点。
获得zip包的body后还得进行解压。请添加图片描述

先来个流程图
Created with Raphaël 2.3.0 开始解析byte数组 从 下标0 开始读取 4byte , 这是包长度(body长度+header长度)。 从 下标4 开始读取 2byte , 这是header长度。 从 下标6 开始读取 2byte , Version:0、普通数据,2、zip数据 从 下标8 开始读取 4byte , Operation:消息的类型。 从 下标12 开始读取 4byte , 这是保留字段,可以忽略。 判断Version是否为2,zip包 解压Zip数组 从下标16到读取到“body长度” (body长度 = 包长度 - header长度) 判断是否读完byte数组 结束解析 yes no yes no
代码实现

同样以JAVA代码为例
代码依赖了hutool-core,用于zip数组解压

public static void unpack(ByteBuffer byteBuffer){
     int packageLen = byteBuffer.getInt();
     short headLength = byteBuffer.getShort();
     short protVer = byteBuffer.getShort();
     int optCode = byteBuffer.getInt();
     int sequence = byteBuffer.getInt();
     if(3 == optCode){
         System.out.println("这是服务器心跳回复");
     }
     byte[] contentBytes = new byte[packageLen - headLength];
     byteBuffer.get(contentBytes);
     //如果是zip包就进行解包
     if(2 == protVer){
         unpack(ByteBuffer.wrap(ZipUtil.unZlib(contentBytes)));
         return;
     }

     String content = new String(contentBytes, StandardCharsets.UTF_8);
     if(8 == optCode){
         //返回{"code":0}表示成功
         System.out.println("这是鉴权回复:"+content);
     }
     //真正的弹幕消息
     if(5 == optCode){
         System.out.println("真正的弹幕消息:"+content);
         // todo 自定义处理

     }
     //只存在ZIP包解压时才有的情况
     //如果byteBuffer游标 小于 byteBuffer大小,那就证明还有数据
     if(byteBuffer.position() < byteBuffer.limit()){
         unpack(byteBuffer);
     }
 }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1079030.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PowerShell pnpm : 无法加载文件 C:\Users\lenovo\AppData\Roaming\npm\pnpm.ps1

1、右键点击【开始】&#xff0c;打开Windows PowerShell&#xff08;管理员&#xff09; 2、运行命令set-ExecutionPolicy RemoteSigned 3、根据提示&#xff0c;输入A,回车 此时管理员权限已经可以运行pnpm 如果vsCode还报该错误 继续输入 4、右键点击【开始】&#xff0c;打…

如何将gif变成视频?3个转换方法

如何将gif变成视频&#xff1f;没错&#xff0c;GIF是一种动态图片格式&#xff0c;与视频在本质上有所区别。在一些自媒体平台上&#xff0c;我们无法直接分享GIF格式的图片&#xff0c;但可以将其转换为视频格式后再进行分享。因此&#xff0c;当我们想要分享我们喜欢的GIF图…

C# 文件监听FileSystemWatcher

用处 当文件修改后&#xff0c;触发其他操作&#xff0c;例如删除另一个文件夹下的文件等 代码 using System; using System.Collections.Generic; using System.Diagnostics; using System.IO;namespace ConsoleApp6FileSystemWatcher {internal class Program{static void …

【TES720D-KIT】基于复旦微FMQL20S400全国产化ARM开发套件(核心板+底板)

TES720D-KIT是专门针对我司TES720D&#xff08;基于复旦微FMQL20S400的全国产化ARM核心板&#xff09;的一套开发套件&#xff0c;它包含1个TES720D核心板&#xff0c;加上一个TES720D-EXT扩展底板。 FMQL20S400是复旦微电子研制的全可编程融合芯片&#xff0c;在单芯片内集成…

微信小程序在TS模板下引入TDesign组件

介绍 TDesign 是腾讯官方出品的一款微信小程序组件库。本文介绍如何在新建ts空白模板下引入TDesign库 步骤 新建一个空白项目&#xff0c;这里可以选择TS-基础模板 新建项目目录结构如图所示&#xff1a; 注意这里其实小程序的文件都存放在miniprogram文件夹下&#xff0c;…

本文整理了Debian 11在国内的几个软件源。

1&#xff0e;使用说明 一般情况下&#xff0c;将/etc/apt/sources.list文件中Debian默认的软件仓库地址和安全更新仓库地址修改为国内的镜像地址即可&#xff0c;比如将deb.debian.org和security.debian.org改为mirrors.xxx.com&#xff0c;并使用https访问&#xff0c;可使用…

关于神经网络的思考

关于感知机 感知机&#xff08;Perceptron&#xff09;和神经网络&#xff08;Neural Network&#xff09;之间有一定的关系&#xff0c;可以说感知机是神经网络的一个基本组成单元。 感知机&#xff1a; 感知机是一种简单的二分类线性分类器。它接受多个输入&#xff0c;对每…

sanic框架解决多进程共享缓存问题

最近在用sanic框架做项目&#xff0c;今天需要处理多进程共享缓存问题&#xff0c;在网上搜索了很多&#xff0c;知道使用multiprocessing模块&#xff0c;但是导入后&#xff0c;直接使用会报错&#xff0c;然后看官网解决问题。 直接看官方文档点我哦 大致意思如下&#xf…

flutter 常用组件:文本、图片和按钮

文章目录 文本控件富文本控件图片本地图片网络图片按钮文本控件 ##一’码’当先 Text(这是一段文本这是一段文本这是一段文本这是一段文本这是一段文本这是一段文本这是一段文本这是一段文本,textAlign:TextAlign.center,style: TextStyle(fontWeight: FontWeight.bold, font…

简单好用的CHM文件阅读器 CHM Viewer Star最新 for mac

CHM Viewer Star 是一款适用于 Mac 平台的 CHM 文件阅读器软件&#xff0c;支持本地和远程 CHM 文件的打开和查看。它提供了直观易用的界面设计&#xff0c;支持多种浏览模式&#xff0c;如书籍模式、缩略图模式和文本模式等&#xff0c;并提供了丰富的功能和工具&#xff0c;如…

elasticsearch(ES)分布式搜索引擎01——(初识ES,索引库操作和文档操作,RestClient操作索引库和文档)

目录 1.初识elasticsearch1.1.了解ES1.1.1.elasticsearch的作用1.1.2.ELK技术栈1.1.3.elasticsearch和lucene1.1.4.总结 1.2.倒排索引1.2.1.正向索引1.2.2.倒排索引1.2.3.正向和倒排 1.3.es的一些概念1.3.1.文档和字段1.3.2.索引和映射1.3.3.mysql与elasticsearch1.4.3.总结 2.…

mysql面试题41:关心过业务系统里面的sql耗时吗?统计过慢查询吗?对慢查询怎么优化呢?

该文章专注于面试&#xff0c;面试只要回答关键点即可&#xff0c;不需要对框架有非常深入的回答&#xff0c;如果你想应付面试&#xff0c;是足够了&#xff0c;抓住关键点 面试官&#xff1a;关心过业务系统里面的sql耗时吗&#xff1f;统计过慢查询吗&#xff1f;对慢查询怎…

[CSAWQual 2019]Web_Unagi - 文件上传+XXE注入(XML编码绕过)

[CSAWQual 2019]Web_Unagi 1 解题流程1.1 分析1.2 解题 2 思考总结 1 解题流程 这篇博客讲了xml进行编码转换绕过的原理&#xff1a;https://www.shawroot.cc/156.html 1.1 分析 页面可以上传&#xff0c;上传一句话php失败&#xff0c;点击示例发现是xml格式&#xff0c;那…

『Linux小程序』进度条

文章目录 缓冲区问题回车与换行的区别进度条小程序 缓冲区问题 假设有一段代码为: #include<iostream> #include<unistd.h> int main() …

openGauss学习笔记-96 openGauss 数据库管理-访问外部数据库-file_fdw

文章目录 openGauss学习笔记-96 openGauss 数据库管理-访问外部数据库-file_fdw96.1 使用file_fdw96.2 注意事项 openGauss学习笔记-96 openGauss 数据库管理-访问外部数据库-file_fdw openGauss的fdw实现的功能是各个openGauss数据库及远程服务器&#xff08;包括数据库、文件…

算法-DFS+记忆化/动态规划-不同路径 II

算法-DFS记忆化/动态规划-不同路径 II 1 题目概述 1.1 题目出处 https://leetcode.cn/problems/unique-paths-ii 1.2 题目描述 2 DFS记忆化 2.1 思路 注意题意&#xff0c;每次要么往右&#xff0c;要么往下走&#xff0c;也就是说不能走回头路。但是仍有可能走到之前已经…

动态壁纸软件iWall mac中文特色

iWall for mac是一款动态壁纸软件&#xff0c;它可以使用任何格式的漂亮视频(无须转换)&#xff0c;音频(可视化功能)&#xff0c;图片&#xff0c;动画&#xff0c;Flash&#xff0c;gif&#xff0c;swf&#xff0c;程序&#xff0c;网页&#xff0c;网站做为您的动态壁纸&…

gazebo joint 中的type fixed 和continuous的区别

在 Gazebo 中&#xff0c;机器人描述中的元素type属性<joint>指定连接两个链接的关节类型。接头有多种类型&#xff0c;包括fixed和continuous。这是它们之间的区别&#xff1a; 固定关节 ( type“fixed”): 固定关节代表两个链接之间的刚性连接。它不允许连接的链接之…

初识Linux(入门篇)

文章目录 初识 Linux1、Linux 背景1.1、Linux 历史1.2、Linux 官网1.2、Linux 发行版本 2、使用 XShell 远程登录 Linux2.1、下载安装 XShell2.2、查看 Linux 主机 ip2.3、使用 XShell 登陆主机2.4、XShell下的复制粘贴 3、Linux 基本指令3.1、ls 指令3.2、pwd 指令3.3、cd 指令…

iOS 获取模拟器沙盒路径

xcrun simctl get_app_container booted Bundle Identifier data