前言

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

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

本文将包含以下内容:

  1. 我将使用斗鱼TV官方公开的弹幕PDF文档,使用 SocketTcpClient连续斗鱼弹幕;

  2. 分析如何利用 .NET强大的 ValueTask特性,在保持代码简洁的同时,轻松享受高性能异步代码的快乐;

  3. 然后将使用 ReactiveExtensions( RX),演示如何将一系列复杂的弹幕接入操作,就像写 HelloWorld一般容易;

  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.0switchexpression功能,可以一个表达式转成整个颜色转换,比 if/elseswitch/case语句都精简不少,可谓一气呵成。

支持异步/ ValueTaskMemory<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);    }
}

总结

最终运行效果如下:

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

  • 共享本文所使用的所有完整的源代码;

  • 介绍如何使用 ReactiveExtensions( RX),演示这一系列操作用起来,就像写 HelloWorld一样简单;

  • 用我自制的“准游戏引擎” FlysEngine,只需少量代码,即可实现桌面弹幕的效果;

敬请期待!“刷一波666???”

.NET斗鱼直播弹幕客户端(上)相关推荐

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

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

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

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

  3. 斗鱼直播弹幕python_python利用danmu实时获取斗鱼等直播网站字幕

    danmu 是一个开源的直播平台弹幕接口 用python    pip install danmu # -*- coding: utf-8 -*- import time, sys from danm ...

  4. OBS斗鱼直播弹幕插件效果

    我没有安装任何OBS插件,一样达到了美化版弹幕效果

  5. 【干货篇】Vue实现直播弹幕功能

    Vue实现直播弹幕 前言 上一篇文章我们讲述了如何用UniApp和Vue在搭建Nginx直播流媒体服务器后实现直播的功能,那么直播中必不可少的一部分就是弹幕,能够增加直播气氛同时,了解用户对于直播或者 ...

  6. 实时爬取斗鱼直播时的弹幕消息

    最近想爬取实时消息,上网查了下,所以爬取斗鱼直播的弹幕消息做了下练习,这个开源的代码有很多,但是具体是怎么爬取到的还是要仔细地研究下.想爬取斗鱼的弹幕消息,按照常用的做法是打开网页用抓包工具抓包,但是 ...

  7. python 爬取直播弹幕视频_调用斗鱼API爬取直播间弹幕信息(用户昵称及弹幕内容)...

    调用斗鱼API爬取直播间弹幕信息(用户昵称及弹幕内容) 查看<斗鱼弹幕服务器第三方接入协议v1.4.1>,了解斗鱼API的使用方法,即如何连接斗鱼弹幕服务器.维持连接及获取弹幕信息 Pyt ...

  8. 斗鱼直播flash怎么切换html5,GitHub - spacemeowx2/DouyuHTML5Player: 替换斗鱼的Flash弹幕播放器...

    斗鱼HTML5播放器 基于 flv.js 的斗鱼HTML5播放器. 使用了 flv.js 内核提供的直播流播放, 用 JavaScript 实现了斗鱼的弹幕协议, 并支持发送弹幕和送礼物. 使用 不要 ...

  9. python爬斗鱼直播_Python爬虫:利用API实时爬取斗鱼弹幕

    原标题:Python爬虫:利用API实时爬取斗鱼弹幕 这些天一直想做一个斗鱼爬取弹幕,但是一直考试时间不够,而且这个斗鱼的api接口虽然开放了但是我在github上没有找到可以完美实现连接.我看了好多 ...

最新文章

  1. 代码测试意味着完全消灭了Bug?
  2. halcon算子翻译——set_framegrabber_param
  3. frps 多个_frp端口批量穿透教程
  4. wireshark 总体结构
  5. 计算从A地出发到各个地方的路径及距离
  6. 做玫瑰花的方法 用纸_新生活新健康:春饮玫瑰花茶 最是疏肝解郁
  7. 源码分析RocketMQ ACL实现机制
  8. 企业库第4版最后版本下载
  9. linux安装pdo mysql扩展_linux下php安装pdo_mysql扩展
  10. js/jquery 获取本地文件的文件路劲 获取input框中type=‘file’ 中的文件路径(转载)...
  11. Java常用中间件---SpringBoot整合ActivityMQ
  12. 手机内存文件夹html,手机内存越用越小?删除这6个英文文件夹,瞬间释放20G
  13. LTE核心网中,SGW、PGW、PCRF都有计费的功能,请问这几个网元的计费功能有什么区别与联系呢?...
  14. VISUAL SVN安装 及客户端使用
  15. 如何使用WGCLOUD实时监控网站状态并告警
  16. 微软WHQL认证的好处
  17. python文件操作练习题【学生成绩.txt】
  18. 结构化、非结构化和半结构化数据
  19. html 画星星的坐标,如何使用画布HTML5绘制星星?
  20. 爬虫爬取豆瓣电影排行榜

热门文章

  1. 【慢慢学算法】:查找第k小数
  2. 页面上指定类型的控件的样式添加
  3. lvs+keepalived详解
  4. Linux 环境下 jdk1.8 maven3.2.3 Git2.8.0 安装脚本
  5. Sql Server 优化 SQL 查询:如何写出高性能SQL语句
  6. 13.ThreadPoolExecutor线程池之submit方法
  7. Git命令集十三——快照操作
  8. Java里阻塞线程的三种实现方法
  9. .NET6之MiniAPI(二十七):Metrics
  10. 如何使用 .NET Core 安全地加/解密文件