更改此协议以使用 TCP 流式处理

Changing this protocol to work with TCP streaming

本文关键字:TCP 处理 协议      更新时间:2023-10-16

我为我的游戏制作了一个简单的协议:

b = bool
i = int
sINT: = string whose length is INT followed by a : then the string
m = int message id.

例:

m133s11:Hello Worldi-57989b0b1b0

这将是:

Message ID 133
String 'Hello World' length 11
int -57989
bool false
bool true
bool false

但是我不知道TCP可能只能发送消息的一部分。我不确定如何修改它,以便我可以执行以下操作:

on receive data from client:
use client's chunk parser
process data
if has partial message then try to find matching END
if no partial messages then try to read a whole message
for each complete message in queue, dispatch it

我可以通过在消息的开头添加 B 和在末尾添加 E 来做到这一点,然后解析第一个字符为 B,最后一个字符为 E。

唯一的问题是如果我在中间收到一些不遵循协议的愚蠢的东西。或者,如果我应该只收到不是消息而只是一个字符串的东西怎么办。因此,如果我打算以某种方式接收字符串 HelloB,那么我会将其解析为 hello 和消息的开头,但我永远不会收到该消息,因为它不是消息。

如何修改协议以解决这些潜在问题?尽管我预计只会收到格式正确的消息,但如果一个消息编码不佳并且所有内容都失常,那将是一场噩梦。

谢谢

决定在开头添加长度并跟踪我是否正在处理消息:所以:

p32m133s11:Hello Worldi-57989b0b1b0

然后我有 3 种状态,读取以查找"p",读取以查找"p"之后的长度或读取字节直到读取长度字节。

你觉得怎么样?

它似乎工作得很好。

你正在做的是相当老派的磁带东西。好。

您可能遇到的问题是,如果收到消息的一部分,则无法判断您是否正在执行令牌。

例如,如果您收到:

m12
这是消息 12

,还是消息 122 的第一部分?

如果您收到:

i-12
这是整数 -

12 还是整数 -124354 的第一部分?

所以我认为您需要更改它,以便消息编号是固定宽度(例如四位数字),字符串长度是固定的(例如 6 位数字),整数宽度固定为 10 位。

所以你的例子是:

m_133s____11:Hello Worldi____-57989b0b1b0

这样,如果您收到消息的第一部分,则可以存储它并等待收到其余部分,然后再处理它。

您还可以考虑使用控制字符来分隔消息部分。有ASCII控制代码通常用于此目的,RS,FS,GS和US。所以一条消息可能是

[RS]FieldName[US]FieldValue[RS]fieldName[US]FieldValue[GS].

您知道何时有完整的消息,因为[GS]标记结束。然后,您可以使用 [RS] 作为分隔符将其划分为字段,并使用 [US] 将每个字段拆分为名称/值。

有关简短的信息,请参阅 http://en.wikipedia.org/wiki/C0_and_C1_control_codes。