面向对象的网络

时间:2011-02-08 19:14:59

标签: c++ oop networking packets

我已经编写了许多网络系统,并且很清楚网络的工作原理。但是我总是最终拥有一个数据包接收功能,这是一个巨大的switch语句。这开始转向我。我宁愿采用一种优雅的面向对象的方式来处理接收数据包,但每当我尝试提出一个好的解决方案时,我总会遇到短暂的问题。

例如,假设您有一台网络服务器。它只是在那里等待回应。数据包进入,服务器需要验证数据包,然后需要决定如何处理它。

目前我通过在标头中打开数据包ID然后拥有一大堆处理每种数据包类型的函数调用来实现此目的。对于复杂的网络系统,这会产生一个单一的switch语句,我真的不喜欢这样处理它。我考虑过的一种方法是使用处理程序类的映射。然后我可以将数据包传递给相关的类并处理传入的数据。我遇到的问题是我需要一些方法来使用地图“注册”每个数据包处理程序。这意味着,通常,我需要创建类的静态副本,然后在构造函数中将其注册到中央数据包处理程序。虽然这种方法很有效,但它似乎是一种处理它的不雅和繁琐的方式。

编辑:同样,拥有一个兼顾两种方式的漂亮系统也是理想的选择。即一个类结构,它容易处理发送相同的数据包类型(显然通过不同的功能)。

有人能指出我更好的处理传入数据包的方法吗?非常感谢链接和有用的信息!

道歉,如果我没有很好地描述我的问题,因为我无法很好地描述它也是我从未设法提出解决方案的原因。

7 个答案:

答案 0 :(得分:5)

关于处理数据包类型的方式:对我来说,地图是最好的。但是我使用普通数组(或矢量)而不是地图。如果从0开始按顺序枚举数据包类型,它将使访问时间保持不变。

关于班级结构。有些图书馆已经完成了这项工作:Available Game network protocol definition languages and code generation。例如。 Google's Protocol Buffer似乎很有希望。它为协议描述中的每条消息生成一个包含getter,setter,serialization和deserialized例程的存储类。协议描述语言或多或少看起来很丰富。

答案 1 :(得分:1)

处理程序实例的映射几乎是处理它的最佳方法。没有什么不优雅的。

答案 2 :(得分:1)

根据我的经验,表驱动的解析是最有效的方法。

虽然std::map很好,但最终我还是使用了静态表。 std::map无法静态初始化为常量表。它必须在运行时加载。表(结构数组)可以声明为数据并在编译时初始化。我没有遇到过线性搜索成为瓶颈的表格。通常表大小足够小,以至于二进制搜索的开销比线性搜索慢。

为了获得高性能,我将使用消息数据作为表的索引。

答案 3 :(得分:1)

当你在做OOP时,你试图将每件事物都表示为一个对象,对吧?所以你的协议消息也成了对象;你可能会有一个基类YourProtocolMessageBase,它将封装任何消息的行为,并从中继承你的多态专用消息。然后你只需要一种方法将每个消息(即每个YourProtocolMessageBase实例)转换为一个字节串,以及一种反向的方法。这种方法称为serialization技术;存在一些基于元编程的implementations

Python中的简单示例:

from socket import *
sock = socket(AF_INET6, SOCK_STREAM)
sock.bind(("localhost", 1234))
rsock, addr = sock.accept()

服务器阻止,为客户端启动另一个实例:

from socket import *
clientsock = socket(AF_INET6, SOCK_STREAM)
clientsock.connect(("localhost", 1234))

现在使用Python的内置序列化模块pickle;客户端:

import pickle
obj = {1: "test", 2: 138, 3: ("foo", "bar")}
clientsock.send(pickle.dumps(obj))

服务器:

>>> import pickle
>>> r = pickle.loads(rsock.recv(1000))
>>> r
{1: 'test', 2: 138, 3: ('foo', 'bar')}

因此,正如您所看到的,我只是通过链接本地发送了一个Python 对象。这不是OOP吗?

我认为序列化唯一可行的替代方法是维护bimapID⇔类。这看起来真的不可避免。

答案 4 :(得分:1)

您希望继续使用相同的数据包网络协议,但在编程中将其转换为对象,对吗?

有几种协议允许您将数据视为编程对象,但似乎您不希望更改协议,只是在应用程序中处理它的方式。

数据包是否带有“标签”或元数据或任何“id”或“数据类型”,可以映射到特定的对象类?如果是,您可以创建一个存储id的数组。和匹配类,并生成一个对象。

答案 5 :(得分:1)

处理此问题的更多方法是使用状态模式构建状态机。

处理传入的原始数据正在解析状态机提供优雅解决方案的位置(您必须在优雅和性能之间进行选择)

你有一个要处理的数据缓冲区,每个状态都有一个句柄缓冲区方法,用于解析和处理他的部分缓冲区(如果已经可以),并根据内容设置下一个状态。

如果你想要提高性能,你仍然可以使用状态机,但不要使用OO部分。

答案 6 :(得分:1)

我将使用Flatbuffers和/或Cap’n Proto代码生成器。