现在直播平台由于弹幕的存在,主播与观众可以更轻松地进行互动,非常受年轻群众的欢迎。斗鱼TV就是一款非常流行的直播平台,弹幕更是非常火爆。看到有不少主播接入弹幕语音播报器弹幕点歌等模块,这都需要首先连接斗鱼弹幕。

经常看到其它编程语言的开发者,分享了他们斗鱼弹幕客户端的代码。.NET当然也能做,还能做得更好(只是不知为何很少见人分享)。

本文将包含以下内容:

  1. 我将使用斗鱼TV官方公开的弹幕PDF文档,使用Socket/TcpClient连续斗鱼弹幕;
  2. 分析如何利用.NET强大的ValueTask特性,在保持代码简洁的同时,轻松享受高性能异步代码的快乐;
  3. 然后将使用Reactive ExtensionsRX),演示如何将一系列复杂的弹幕接入操作,就像写Hello World一般容易;
  4. 用我自制的“准游戏引擎”FlysEngine,只需少量代码,即可将斗鱼TV的弹幕显示左右飞过的效果;

本文内容可能比较多,因此分上、下两篇阐述,上篇将具体聊聊第1、2点,第3、4点将在下篇进行,整篇完成后,最终效果如下:

斗鱼直播API

现在网上可以轻松找到斗鱼弹幕服务器第三方接入协议v1.6.2.pdf(网上搜索该关键字即可找到)。

文档提到,第三方接入弹幕服务的服务器为openbarrage.douyutv.com:8601,我们可以使用TcpClient来方便连接:

using (var client = new TcpClient())
{
client.ConnectAsync("openbarrage.douyutv.com", 8601).Wait();
Stream stream = client.GetStream();
// do other works
}

该文档中提到所有数据包格式如下:

注意前两个4字节的消息长度是完全一样的,可以使用Debug.Assert进行断言。

其中所有数字都为小端整数,刚好.NETBinaryWriter类默认都以小端整数进行转换。可以利用起来。

因此,读取一个消息包的完整代码如下:

using (var reader = new BinaryReader(stream, Encoding.UTF8, true))
{
var fullMsgLength = reader.ReadInt32();
var fullMsgLength2 = reader.ReadInt32();
Debug.Assert(fullMsgLength == fullMsgLength2); var length = fullMsgLength - 1 - 4 - 4;
var packType = reader.ReadInt16();
Debug.Assert(packType == ServerSendToClient);
var encrypted = reader.ReadByte();
Debug.Assert(encrypted == Encrypted);
var reserved = reader.ReadByte();
Debug.Assert(reserved == Reserved); var bytes = reader.ReadBytes(length);
var zero = reader.ReadByte();
Debug.Assert(zero == ByteZero);
}

其中bytes既是数据部分,根据pdf文档中的规定,该部分为UTF-8编码,在C#中使用Encoding.UTF8.GetString()即可获取其字符串,该字符串长这样子:

type@=chatmsg/rid@=633019/ct@=1/uid@=124155/nn@=夜科扬羽/txt@=这不压个蜥蜴/cid@=602c7f1becf2419962a6520300000000/ic@=avatar@S000@S12@S41@S55_avatar/level@=21/sahf@=0/cst@=1570891500125/bnn@=賊开心/bl@=8/brid@=5789561/hc@=21ebd5b2c86c01e0565453e45f14ca5b/el@=/lk@=/urlev@=10/

该格式不是JSON/XML等,但仔细分析又确实有逻辑,有层次感,根据文档,该格式为所谓的STT序列化,该格式包含键值对、数组等多种格式。相比JSON可以减少大量的引号"空间开销。还好协议简单,我可以通过寥寥几行代码,即可转换为Json.NETJToken格式:

public static JToken DecodeStringToJObject(string str)
{
if (str.Contains("//")) // 数组
{
var result = new JArray();
foreach (var field in str.Split(new[] { "//" }, StringSplitOptions.RemoveEmptyEntries))
{
result.Add(DecodeStringToJObject(field));
}
return result;
}
if (str.Contains("@=")) // 对象
{
var result = new JObject();
foreach (var field in str.Split(new[] { '/' }, StringSplitOptions.RemoveEmptyEntries))
{
var tokens = field.Split(new[] { "@=" }, StringSplitOptions.None);
var k = tokens[0];
var v = UnscapeSlashAt(tokens[1]);
result[k] = DecodeStringToJObject(v);
}
return result;
}
else if (str.Contains("@A=")) // 键值对
{
return DecodeStringToJObject(UnscapeSlashAt(str));
}
else
{
return UnscapeSlashAt(str); // 值
}
} static string EscapeSlashAt(string str)
{
return str
.Replace("/", "@S")
.Replace("@", "@A");
} static string UnscapeSlashAt(string str)
{
return str
.Replace("@S", "/")
.Replace("@A", "@");
}

这样一来,即可将STT格式转换为JSON格式,因此只需像JSON格式取出nn字段和txt字段即可,还有一个col字段,可以用来确定弹幕颜色,我可以将其转换为RGBint32值:

Color = (x["col"] ?? new JValue(0)).Value<int>() switch
{
1 => 0xff0000, // 红
2 => 0x1e87f0, // 浅蓝
3 => 0x7ac84b, // 浅绿
4 => 0xff7f00, // 橙色
5 => 0x9b39f4, // 紫色
6 => 0xff69b4, // 洋红
_ => 0xffffff, // 默认,白色
}

该代码使用了C# 8.0switch expression功能,可以一个表达式转成整个颜色转换,比if/elseswitch/case语句都精简不少,可谓一气呵成。

支持异步/ValueTask/Memory<T>优化

C# 5.0提供了强大的异步API——async/await,通过异步API,以前难以用编程实现的操作现在可以像写串行代码一样轻松完成,还能轻松加入取消任务操作。

然后C# 7.0发布了ValueTaskValueTask是值类型,因此在频繁调用异步操作(如使用Stream读取字节)时,不会因为创建过多的Task而分配没必要的内存。这里,我确实是使用TCP连接流读取字节,是使用ValueTask的最佳时机。

这里我们将尝试将代码切换为ValueTask版本。

首先第一个问题是BinaryReader类,该类提供了便利的字节操作方式,且能确保字节端为小端,但该类不提供异步API,因此需要作一些特殊处理:

public static async Task<string> RecieveAsync(Stream stream, CancellationToken cancellationToken)
{
int fullMsgLength = await ReadInt32().ConfigureAwait(false);
int fullMsgLength2 = await ReadInt32().ConfigureAwait(false);
Debug.Assert(fullMsgLength == fullMsgLength2); int length = fullMsgLength - 1 - 4 - 4;
short packType = await ReadInt16().ConfigureAwait(false);
Debug.Assert(packType == ServerSendToClient);
short encrypted = await ReadByte().ConfigureAwait(false);
Debug.Assert(encrypted == Encrypted);
short reserved = await ReadByte().ConfigureAwait(false);
Debug.Assert(reserved == Reserved); Memory<byte> bytes = await ReadBytes(length).ConfigureAwait(false);
byte zero = await ReadByte().ConfigureAwait(false);
Debug.Assert(zero == ByteZero); return Encoding.UTF8.GetString(bytes.Span);
}

如代码所示,我封装了ReadInt16()ReadInt32()两个方法,

var intBuffer = new byte[4];
var int32Buffer = new Memory<byte>(intBuffer, 0, 4); async ValueTask<int> ReadInt32()
{
var memory = int32Buffer;
int read = 0;
while (read < 4)
{
read += await stream.ReadAsync(memory.Slice(read), cancellationToken).ConfigureAwait(false);
}
Debug.Assert(read == memory.Length);
return
(intBuffer[0] << 0) +
(intBuffer[1] << 8) +
(intBuffer[2] << 16) +
(intBuffer[3] << 24);
}

如图,我还使用了一个while语句,因为不像BinaryReader,如果一次无法读取所需的字节数(4个字节),stream.ReadAsync()并不会堵塞线程。然后需要将int32Buffer转换为int类型。

注意:此处我没有使用BitConverter.ToInt32(),也不能使用该方法,因为该方法不像BinaryReader,它在大端/小端的CPU上会有不同的行为。(其中在大端CPU上将有错误的行为)涉及二进制序列化需要传输的,不能使用BitConverter类。

同样的,写TCP流也需要有相应的变化:

static async Task SendAsync(Stream stream, byte[] body, CancellationToken cancellationToken)
{
var buffer = new byte[4]; await stream.WriteAsync(GetBytesI32(4 + 4 + body.Length + 1), cancellationToken).ConfigureAwait(false);
await stream.WriteAsync(GetBytesI32(4 + 4 + body.Length + 1), cancellationToken).ConfigureAwait(false); await stream.WriteAsync(GetBytesI16(ClientSendToServer), cancellationToken).ConfigureAwait(false);
await stream.WriteAsync(new byte[] { Encrypted}, cancellationToken).ConfigureAwait(false);
await stream.WriteAsync(new byte[] { Reserved}, cancellationToken).ConfigureAwait(false); await stream.WriteAsync(body, cancellationToken).ConfigureAwait(false);
await stream.WriteAsync(new byte[] { ByteZero}, cancellationToken).ConfigureAwait(false); Memory<byte> GetBytesI32(int v)
{
buffer[0] = (byte)v;
buffer[1] = (byte)(v >> 8);
buffer[2] = (byte)(v >> 16);
buffer[3] = (byte)(v >> 24);
return new Memory<byte>(buffer, 0, 4);
} Memory<byte> GetBytesI16(short v)
{
buffer[0] = (byte)v;
buffer[1] = (byte)(v >> 8);;
return new Memory<byte>(buffer, 0, 2);
}
}

总结

最终运行效果如下:

这一篇文章介绍了如何使用斗鱼tv开放弹幕API,下篇将会:

  • 共享本文所使用的所有完整的源代码;
  • 介绍如何使用Reactive ExtensionsRX),演示这一系列操作用起来,就像写Hello World一样简单;
  • 用我自制的“准游戏引擎”FlysEngine,只需少量代码,即可实现桌面弹幕的效果;

敬请期待!“刷一波666”

喜欢的朋友请关注我的微信公众号:【DotNet骚操作】

.NET斗鱼直播弹幕客户端(上)的更多相关文章

  1. .NET斗鱼直播弹幕客户端(下)

    .NET斗鱼直播弹幕客户端(下) 在上篇文章中,我们提到了如何使用.NET连接斗鱼TV直播弹幕的基本操作.然而想要做得好,做得容易扩展,就需要做进一步的代码整理. 本文将涉及以下内容: 介绍如何使用R ...

  2. .NET斗鱼直播弹幕客户端(2021)

    .NET斗鱼直播弹幕客户端(2021) 离之前更新的两篇<.NET斗鱼直播弹幕客户端>已经有一段时间,近期有许多客户向我反馈刚好有这方面的需求,但之前的代码不能用了--但网上许多流传的No ...

  3. .NET斗鱼直播弹幕客户端(上)

    现在直播平台由于弹幕的存在,主播与观众可以更轻松地进行互动,非常受年轻群众的欢迎.斗鱼TV就是一款非常流行的直播平台,弹幕更是非常火爆.看到有不少主播接入弹幕语音播报器.弹幕点歌等模块,这都需要首先连 ...

  4. android文件管理器源码、斗鱼直播源码、企业级erp源码等

    Android精选源码 文件清理管理器 自定义水平带数字的进度条以及自定义圆形带数字的进度条 利用sectionedRecyclerViewAdapter实现分组列表的recyclerView源码 流 ...

  5. Android Studio 直播弹幕

    我只是搬运:https://blog.csdn.net/HighForehead/article/details/55520199 写的很好很详细,挺有参考价值的 demo直通车:https://do ...

  6. ubuntu下使用OBS开斗鱼直播

    系统环境:ubuntu 15.10,OBS Studio 0.13.1 OBS是可以在linux,windows,mac下直播的开源软件,官方地址:https://obsproject.com/ 斗鱼 ...

  7. Scrapy项目 - 实现斗鱼直播网站信息爬取的爬虫设计

    要求编写的程序可爬取斗鱼直播网站上的直播信息,如:房间数,直播类别和人气等.熟悉掌握基本的网页和url分析,同时能灵活使用Xmind工具对Python爬虫程序(网络爬虫)流程图进行分析.   一.项目 ...

  8. 直播弹幕抓取逆向分析流程总结 websocket,flash

    前端无秘密 直播的逆向抓取说到底是前端的调试和逆向技术,加上部分的dpa(深入包分析,个人能力尚作不到深入,只能作简单分析)难度较低 目前互联网直播弹幕主要是两种技术实现. 1websocket消息通 ...

  9. iOS仿QQ侧滑菜单、登录按钮动画、仿斗鱼直播APP、城市选择器、自动布局等源码

    iOS精选源码 QQ侧滑菜单,右滑菜单,QQ展开菜单,QQ好友分组 登录按钮 3分钟快捷创建高性能轮播图 ScrollView嵌套ScrolloView(UITableView .UICollecti ...

  10. asp.net 客户端上传文件全路径获取方法

    asp.net  获取客户端上传文件全路径方法: eg:F:\test\1.doc 基于浏览器安全问题,浏览器将屏蔽获取客户端文件全路径的方法,只能获取到文件的文件名,如果需要获取全路径则需要另想其他 ...

随机推荐

  1. Linux离线安装Tomcat

    系统环境: centos7.3.1611 openjdk version "1.8.0_102" apache-tomcat-9.0.36.tar.gz tomcat 安装 #链接 ...

  2. 使用 reloadNuxtApp 强制刷新 Nuxt 应用

    title: 使用 reloadNuxtApp 强制刷新 Nuxt 应用 date: 2024/8/22 updated: 2024/8/22 author: cmdragon excerpt: re ...

  3. AWS EC2 实例类型命名规则

    AWS EC2(Elastic Compute Cloud)实例类型的命名规则反映了实例的性能特征.用途和硬件配置.这些实例类型的名称由几个组件构成,每个组件都提供了关于该实例类型特定方面的信息.理解 ...

  4. sql server create table 给字段添加注释说明

    EXEC sys.sp_addextendedproperty @name=N'MS_Description',@level1name=N'a_jcgl_data',@level2name=N'id' ...

  5. 光影精灵10 Win1+Ubuntu18.04 双系统 踩坑记录

    前言 第二年准备报名智能车了,当然还是创意组别.刚好买了今年新出的电脑光影精灵10,我想着也给它安一个双系统.但是没想到,相比于之前那个老电脑,新电脑的新硬件和驱动问题远比老电脑麻烦的多. 在经历了一 ...

  6. 痞子衡嵌入式:在MDK开发环境下自定义安装与切换不同编译器版本的方法

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是在MDK开发环境下自定义安装与切换不同编译器版本的方法. Keil MDK 想必是嵌入式开发者最熟悉的工具之一了,自 2005 年 Ar ...

  7. TFC-Pretraining: 基于时间频率一致性对时间序列进行自监督对比预训练《Self-Supervised Contrastive Pre-Training for Time Series via Time-Frequency Consistency》(时间序列、时序表征、时频一致性、对比学习、自监督学习)

    2023年11月10日,今天看一篇论文,现在17:34,说实话,想摆烂休息,不想看,可还是要看,拴Q. 论文:Self-Supervised Contrastive Pre-Training for ...

  8. Angular 18+ 高级教程 – 目录

    请按顺序阅读 关于本教程 初识 Angular Get Started Angular Compiler (AKA ngc) Quick View Dependency Injection 依赖注入  ...

  9. 平面设计 – 色轮 & 配色

    前言 由于之前那篇有点长, 而且色轮很重要, 所以独立写一篇呗. 参考: 一文看懂色轮 Youtube – 03 色彩 (什么是色相.纯度.明度.色环.补色?怎样配色?) Youtube – 初學繪畫 ...

  10. Clickhouse-insert 数据写入不成功问题

    [应用场景] 对副本表进行 alter delete 数据后,同样的数据再进行 insert into 操作. [问题复现] [问题解释] 对副本表 insert 语句的数据会划分为数据块. 每个数据 ...