- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我想设计一个应用程序协议(protocol),我在其中定义了一个名为 message 的类,它具有 4 个字节的固定 header 大小和可变负载部分。要将负载分配给消息,我想按以下方式执行:
message msg;
IPAddressv4 ipv4("127.0.0.1");
IPAddressv6 ipv6("::1");
Algorithm alg(2);
msg.version(1);
msg.ack(1);
msg << ipv4 & ipv6 & alg;
其中 IPAddressv4、IPAddressv6 和 Algorithm 是具有基本数据类型的定义对象。例如 ipv4 是 uint32,ipv6 是 2*unit64,算法是 uint8。
这样最终的消息大小为:固定 header 大小“4”+ 4 + 16 + 1 = 25 字节
那么谁能告诉我如何在 C++ 中实现它的一般方法,或者无论如何在 boost 库中都可以做到这一点?
最佳答案
因此,我认为您需要一个行为类似于流但具有二进制格式的类。
这是我为此类(简化)的简单界面:
struct Message
{
// serialize all of the inserted buffer content to the given stream
friend std::ostream& operator <<(std::ostream& os, Message const& msg);
// read all of the given stream into the message buffer for extraction
friend std::istream& operator >>(std::istream& is, Message const& msg);
// insert integral data into the buffer, with fixed endianness
template <typename Int>
friend Message& operator <<(Message& msg, Int v);
template <typename Int, size_t N>
friend Message& operator <<(Message& msg, Int const (&v)[N]);
// extract integral data from the buffer, with fixed endianness
template <typename Int>
friend Message& operator >>(Message& msg, Int& v);
template <typename Int, size_t N>
friend Message& operator >>(Message& msg, Int (&v)[N]);
// default construction
Message() : buffer { "\xee\x33" }
{
buffer.unsetf(std::ios::skipws);
}
private:
std::stringstream mutable buffer;
};
我省略了处理 SFINAE/元编程的神秘细节。构造函数快速而肮脏,并展示了一种拥有固定消息 header (在本例中为 ee33
)的简单方法。
现在你可以像这样使用它:
std::string serialize_message()
{
uint8_t ipv4[4] = { 127, 0, 0, 1 };
uint64_t ipv6[2] = { 0xdeadbeef00004b1d, 0xcafed00dcafebabe };
uint8_t alg = 2;
Message msg;
msg << ipv4 << ipv6 << alg;
// stringyfy
std::stringstream ss;
ss << msg;
return ss.str();
}
int main()
{
std::string const rep = serialize_message();
std::cout << rep << std::flush; // for demo (view in hex)
// reconstruct
Message roundtrip;
std::istringstream(rep) >> roundtrip;
uint8_t ipv4[4] = { 0 };
uint64_t ipv6[2] = { 0 };
uint8_t alg = 0;
roundtrip >> ipv4 >> ipv6 >> alg;
assert(rep == boost::lexical_cast<std::string>(roundtrip));
}
请注意 lexical_cast
仅使用上面列出的流插入运算符。
我使用 Boost Spirit 的二进制文件 parsers 实现了这个和 generators .
输入类型到解析器/生成器类型的映射在详细命名空间中。
查看 Live On Coliru 。输出:
0000000: ee33 7f00 0001 dead beef 0000 4b1d cafe .3..........K...
0000010: d00d cafe babe 02 .......
#define LIVE_DANGEROUSLY
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/karma.hpp>
#include <boost/integer.hpp>
#include <sstream>
namespace detail
{
using namespace boost::spirit;
// meta programming helpers
template <int bits, typename endianness = struct big> struct int_traits;
template <typename endianness> struct int_traits<8, endianness/*ignored*/> {
typedef uint8_t type;
typedef qi::byte_type parser;
typedef karma::byte_type generator;
};
template <> struct int_traits<16, struct big> {
typedef uint16_t type;
typedef qi::big_word_type parser;
typedef karma::big_word_type generator;
};
template <> struct int_traits<32, struct big> {
typedef uint32_t type;
typedef qi::big_dword_type parser;
typedef karma::big_dword_type generator;
};
template <> struct int_traits<64, struct big> {
typedef uint64_t type;
typedef qi::big_qword_type parser;
typedef karma::big_qword_type generator;
};
template <> struct int_traits<16, struct little> {
typedef uint16_t type;
typedef qi::little_word_type parser;
typedef karma::little_word_type generator;
};
template <> struct int_traits<32, struct little> {
typedef uint32_t type;
typedef qi::little_dword_type parser;
typedef karma::little_dword_type generator;
};
template <> struct int_traits<64, struct little> {
typedef uint64_t type;
typedef qi::little_qword_type parser;
typedef karma::little_qword_type generator;
};
}
struct Message
{
friend std::ostream& operator <<(std::ostream& os, Message const& msg) {
msg.buffer.seekg(0, std::ios_base::beg);
return os << msg.buffer.rdbuf();
}
friend std::istream& operator >>(std::istream& is, Message const& msg) {
msg.buffer.clear(); // this will overwrite our header
msg.buffer.str("");
return is >> msg.buffer.rdbuf();
}
template <typename Int>
friend typename boost::enable_if<typename boost::is_integral<Int>::type, Message&>::type
operator <<(Message& msg, Int v)
{
msg.buffer.seekp(0, std::ios_base::end);
msg.append_int(v);
return msg;
}
template <typename Int, size_t N>
friend typename boost::enable_if<typename boost::is_integral<Int>::type, Message&>::type
operator <<(Message& msg, Int const (&v)[N])
{
msg.buffer.seekp(0, std::ios_base::end);
msg.append_int(v);
return msg;
}
template <typename Int>
friend typename boost::enable_if<typename boost::is_integral<Int>::type, Message&>::type
operator >>(Message& msg, Int& v)
{
msg.extract_int(v);
return msg;
}
template <typename Int, size_t N>
friend typename boost::enable_if<typename boost::is_integral<Int>::type, Message&>::type
operator >>(Message& msg, Int (&v)[N])
{
msg.extract_ints(v);
return msg;
}
Message() : buffer { "\xee\x33" }
{
buffer.unsetf(std::ios::skipws);
}
private:
std::stringstream mutable buffer;
template <typename Int, int bits=std::numeric_limits<Int>::digits + std::numeric_limits<Int>::is_signed>
void append_int(Int v)
{
namespace karma = boost::spirit::karma;
static const typename detail::int_traits<bits>::generator gen {};
karma::generate(std::ostreambuf_iterator<char>(buffer), gen, static_cast<typename detail::int_traits<bits>::type>(v));
}
template <typename Int, int bits=std::numeric_limits<Int>::digits + std::numeric_limits<Int>::is_signed>
void extract_int(Int& v)
{
namespace qi = boost::spirit::qi;
static const typename detail::int_traits<bits>::parser p {};
boost::spirit::istream_iterator f(buffer), l;
qi::parse(f, l, p, v);
}
template <typename Int, size_t N, int bits=std::numeric_limits<Int>::digits + std::numeric_limits<Int>::is_signed>
void append_int(Int const(&v)[N])
{
#ifdef LIVE_DANGEROUSLY // optimize, but implementation defined cast
namespace karma = boost::spirit::karma;
static const typename detail::int_traits<bits>::generator gen {};
auto data = reinterpret_cast<typename detail::int_traits<bits>::type const*>(&v[0]);
karma::generate(std::ostreambuf_iterator<char>(buffer), +gen, boost::make_iterator_range(data,data+N));
#else
for(auto& i:v)
append_int(i);
#endif
}
template <typename Int, size_t N, int bits=std::numeric_limits<Int>::digits + std::numeric_limits<Int>::is_signed>
void extract_ints(Int (&v)[N])
{
for(auto& i:v)
extract_int(i);
}
};
std::string serialize_message()
{
uint8_t ipv4[4] = { 127, 0, 0, 1 };
uint64_t ipv6[2] = { 0xdeadbeef00004b1d, 0xcafed00dcafebabe };
uint8_t alg = 2;
Message msg;
msg << ipv4 << ipv6 << alg;
// stringyfy
std::stringstream ss;
ss << msg;
return ss.str();
}
int main()
{
std::string const rep = serialize_message();
std::cout << rep << std::flush; // for demo (view in hex)
// reconstruct
Message roundtrip;
std::istringstream(rep) >> roundtrip;
uint8_t ipv4[4] = { 0 };
uint64_t ipv6[2] = { 0 };
uint8_t alg = 0;
roundtrip >> ipv4 >> ipv6 >> alg;
assert(rep == boost::lexical_cast<std::string>(roundtrip));
}
关于c++ - 将类序列化为一个定义的对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23460550/
我的一位教授给了我们一些考试练习题,其中一个问题类似于下面(伪代码): a.setColor(blue); b.setColor(red); a = b; b.setColor(purple); b
我似乎经常使用这个测试 if( object && object !== "null" && object !== "undefined" ){ doSomething(); } 在对象上,我
C# Object/object 是值类型还是引用类型? 我检查过它们可以保留引用,但是这个引用不能用于更改对象。 using System; class MyClass { public s
我在通过 AJAX 发送 json 时遇到问题。 var data = [{"name": "Will", "surname": "Smith", "age": "40"},{"name": "Wil
当我尝试访问我的 View 中的对象 {{result}} 时(我从 Express js 服务器发送该对象),它只显示 [object][object]有谁知道如何获取 JSON 格式的值吗? 这是
我有不同类型的数据(可能是字符串、整数......)。这是一个简单的例子: public static void main(String[] args) { before("one"); }
嗨,我是 json 和 javascript 的新手。 我在这个网站找到了使用json数据作为表格的方法。 我很好奇为什么当我尝试使用 json 数据作为表时,我得到 [Object,Object]
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我听别人说 null == object 比 object == null check 例如: void m1(Object obj ) { if(null == obj) // Is thi
Match 对象 提供了对正则表达式匹配的只读属性的访问。 说明 Match 对象只能通过 RegExp 对象的 Execute 方法来创建,该方法实际上返回了 Match 对象的集合。所有的
Class 对象 使用 Class 语句创建的对象。提供了对类的各种事件的访问。 说明 不允许显式地将一个变量声明为 Class 类型。在 VBScript 的上下文中,“类对象”一词指的是用
Folder 对象 提供对文件夹所有属性的访问。 说明 以下代码举例说明如何获得 Folder 对象并查看它的属性: Function ShowDateCreated(f
File 对象 提供对文件的所有属性的访问。 说明 以下代码举例说明如何获得一个 File 对象并查看它的属性: Function ShowDateCreated(fil
Drive 对象 提供对磁盘驱动器或网络共享的属性的访问。 说明 以下代码举例说明如何使用 Drive 对象访问驱动器的属性: Function ShowFreeSpac
FileSystemObject 对象 提供对计算机文件系统的访问。 说明 以下代码举例说明如何使用 FileSystemObject 对象返回一个 TextStream 对象,此对象可以被读
我是 javascript OOP 的新手,我认为这是一个相对基本的问题,但我无法通过搜索网络找到任何帮助。我是否遗漏了什么,或者我只是以错误的方式解决了这个问题? 这是我的示例代码: functio
我可以很容易地创造出很多不同的对象。例如像这样: var myObject = { myFunction: function () { return ""; } };
function Person(fname, lname) { this.fname = fname, this.lname = lname, this.getName = function()
任何人都可以向我解释为什么下面的代码给出 (object, Object) 吗? (console.log(dope) 给出了它应该的内容,但在 JSON.stringify 和 JSON.parse
我正在尝试完成散点图 exercise来自免费代码营。然而,我现在只自己学习了 d3 几个小时,在遵循 lynda.com 的教程后,我一直在尝试确定如何在工具提示中显示特定数据。 This code
我是一名优秀的程序员,十分优秀!