可以将结构体转换为另一个结构体吗?

14

有人可以描述一下这里是如何使用 (struct sockaddr *)&server 的吗?是否可能将更大的结构转换为较小的结构?

看看这些结构:

// IPv4 AF_INET sockets:
struct sockaddr_in {
    short            sin_family;   // e.g. AF_INET, AF_INET6
    unsigned short   sin_port;     // e.g. htons(3490)
    struct in_addr   sin_addr;     // see struct in_addr, below
    char             sin_zero[8];  // zero this if you want to
};

struct in_addr {
    unsigned long s_addr;          // load with inet_pton()
};

struct sockaddr {
    unsigned short    sa_family;    // address family, AF_xxx
    char              sa_data[14];  // 14 bytes of protocol address
};

这是主程序:

int main(int argc , char *argv[])
 {
        int socket_desc;
        struct sockaddr_in server;

    //Create socket
    socket_desc = socket(AF_INET , SOCK_STREAM , 0);
    if (socket_desc == -1)
    {
        printf("Could not create socket");
    }

    server.sin_addr.s_addr = inet_addr("74.125.235.20");
    server.sin_family = AF_INET;
    server.sin_port = htons( 80 );

    //Connect to remote server
    if (connect(socket_desc , (struct sockaddr *)&server , sizeof(server)) < 0)
    {
        puts("connect error");
        return 1;
    }

    puts("Connected");
    return 0;
}

相关的内容:https://dev59.com/uW865IYBdhLWcg3wFqdV || https://dev59.com/xG865IYBdhLWcg3wivGi - Ciro Santilli OurBigBook.com
4个回答

10

这被称为类型转换(Type Punning)。在这里,两个结构具有相同的大小,因此不存在结构大小的问题。虽然你几乎可以将任何东西强制转换为其他类型,但使用结构进行转换容易出错。


connect() 函数如何访问存储在 struct in_addr 中的 inet_address? - Milad Khajavi
类似于 ((struct sockaddr_in *) server->sin_addr.s_addr) - Benny
printf("地址:%d\n", ((struct sockaddr_in *)server->sin_addr.s_addr)); 会产生以下错误:socket.c:50:54: error: invalid type argument of ‘->’ (have ‘struct sockaddr_in’),为什么? - Milad Khajavi
我的问题已经解决了。谢谢。((struct sockaddr_in *) server)->sin_addr.s_addr - Milad Khajavi

6
这是C语言中的“继承”形式(注意引号)。这种方法的可行性在于C语言并不关心地址的底层数据,而只关心你代表它的方式。
函数通过使用sa_family字段并将其转换为适当的sockaddr_in来确定实际结构体类型。

3

你可以将sockaddr_in转换为sockaddr,但通常情况下你不能将任何一个结构体强制转换为另一个并假定一切都能正常工作。


1
在 C 语言中,可以将任何类型转换为任何其他类型。你甚至可以省略 (struct sockaddr*) 的强制类型转换,但可能会收到编译器的警告。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接