我的目的是构建一个配置文件。因此,我基本上必须以某种方式表示命令包结构。
当实际数据(二进制)到来时,我想将数据包与此配置文件进行比较,然后进行处理(将数据转换为CSV格式)。
所以有不同的命令类型。
因此,每次当数据包到来时,我必须获取它的操作码,并用配置文件检查它,然后返回代表该数据包格式的适当命令格式。
命令格式可能如下所示:
操作码-1字节-整数
命令-4字节-字符串
..。
所有命令都不具有相同数量的字段或相同格式。
我想检索所有这些详细信息。我可以用XML表示它,并使用像libxml2这样的库来解析它。
下面给出了一个示例XML格式:
<cmd type="Multiplication">
<field name="opcode" type="string" bytes="4"/>
<field name="Multiplicand" type="number" bytes="2"/>
<field name="Multiplier" type="number" bytes="2"/>
</cmd>
但这种方法相当缓慢。
我的想法是以某种方式在结构中表示命令包格式。但由于C/C++不是一种反射语言,结构成员是未知的,您需要为每个结构(命令)提供一个函数来解析它。
请建议一些存储格式的方法,以便一个通用函数只需查看此格式即可解析二进制数据。
任何帮助都是非常感谢的。
发布于 2019-05-13 01:18:48
我认为最好的方法是将文件表示为正确命令类型的variant
的集合。例如,假设您有三个命令选项:
struct Constant {
short value;
};
struct UnaryOperation {
unsigned char opcode;
short value;
};
struct BinaryOperation {
unsigned char opcode;
short value1;
short value2;
};
表示未知命令的。如果您有未知命令,则可以将其表示为以下三种类型的variant
:
using Command = std::variant<Constant, UnaryOperation, BinaryOperation>;
根据命令类型应用函数。假设每个命令都有不同的函数:
short eval(Constant c) {
return c.value;
}
short eval(UnaryOperation u) {
switch(u.opcode) {
// stuff
}
}
short eval(BinaryOperation b) {
switch(b.opcode) {
// stuff
}
}
我们可以使用std::visit
来评估任意的Command
short evaluate_command(Command const& command) {
short output;
// This calls the right overload automatically
std::visit(command, [&](auto cmd) { output = eval(cmd); });
return output;
}
解析命令。我们可以从定义的任何类型中自动创建一个std::variant
。这意味着,如果您提供一种方法来确定该命令是基于该文件的,则很容易做到这一点。
enum class OpType : unsigned char {
ConstantOp, UnaryOp, BinaryOp
};
// Command can be automatically constructed from a Constant, a UnaryOperation, or a BinaryOperation
Command readFromStream(std::istream& i) {
OpType type;
unsigned char op;
short value, value2;
// Read the type of the operation
i >> (unsigned char&)type;
//Return either a Constant, a UnaryOperation, or a BinaryOperation
switch(type) {
case OpType::ConstantOp: {
i >> value;
return Constant{value};
}
case OpType::UnaryOp: {
i >> op >> value;
return UnaryOperation{op, value};
}
case OpType::BinaryOp {
i >> op >> value >> value2;
return BinaryOperation{op, value, value2};
}
}
}
https://stackoverflow.com/questions/56100825
复制相似问题