8

考虑我有如下结构:

struct Bitmask
{
  unsigned char payload_length: 7;
  unsigned char mask: 1;
  unsigned char opcode: 4;
  unsigned char rsv3: 1;
  unsigned char rsv2: 1;
  unsigned char rsv1: 1;
  unsigned char fin: 1;
};

const char* payload = "Hello";
const size_t payload_length = strlen(payload);

Bitmask* header = new Bitmask();
header->fin =1;
header->rsv1 = 0;
header->rsv2 = 0;
header->rsv3 = 0;
header->opcode = 1;
header->mask = 0;
header->payload_length = payload_length;

iovec iov[2];
iov[0].iov_base = (char*)header;
iov[0].iov_len = sizeof (header);
iov[1].iov_base = (char *)payload;
iov[1].iov_len = strlen(payload);

ACE_DEBUG ((LM_DEBUG,
            ACE_TEXT ("iov[0].length = %d\niov[1].length = %d\n"),
            iov[0].iov_len,
            iov[1].iov_len));

size_t bytes_xfered;
client_stream_.sendv_n (iov, 2, 0, &bytes_xfered);

cout << "Transfered " << bytes_xfered << " byte(s)" << std::endl;

我正在用适当的值初始化它。最后,我想将结构转换为 char*,这样我就可以附加我的有效负载(即 char* 消息)并通过 websocket 连接发送它。

4

3 回答 3

17

结构的地址是否与其第一个成员的地址相同?

是的,这实际上是 C 和 C++ 标准规定的。来自 C 标准:

6.7.2.1-13。指向结构对象的指针,经过适当转换,指向其初始成员

你的大小struct应该是两个字节。但是,您不应该将指向它的指针转换为char*:相反,您应该使用将您的指针memcpy复制Bitmask到您通过网络发送的缓冲区中。

编辑由于您使用 scatter-gather I/O iovec,您不需要强制Bitmask转换为任何内容:iov_baseis void*,因此您可以简单地设置iov[0].iov_base = header;

注意:这仅在您struct不包含虚函数、基类等时才有效(感谢 Timo)。

编辑2

为了在您的 中获得 {0x81, 0x05} struct,您应该更改结构元素的顺序,如下所示:

struct Bitmask {
    unsigned char opcode: 4; 
    unsigned char rsv3: 1; 
    unsigned char rsv2: 1; 
    unsigned char rsv1: 1; 
    unsigned char fin: 1; 
    unsigned char payload_length: 7; 
    unsigned char mask: 1;
}
于 2012-02-13T01:42:16.620 回答
1

是和不是。

一般来说,这是真的(正如 dasblinkenlight 解释的那样),但它特别不适用于位域。根据 C++11 9.6/3“没有指向位域的指针”,因此它们也没有地址。显然,如果没有“合适的转换” ,“一个指向结构对象的指针,经过适当转换,指向其初始成员”就会崩溃。

于 2012-02-13T12:56:35.377 回答
1

结构的地址与其第一个成员的地址相同,前提是使用了适当的强制转换。鉴于下面的声明struct my_struct,如果item是 struct 类型

my_struct, then (char *)item == &item.wp_cval.


struct my_struct
{

      char wp_cval;
      short wp_font;
      short wp_psize;


}ar[ARSIZE];
于 2012-02-13T13:11:24.667 回答