为什么在调用 bind() 时要将 sockaddr_in 转换为 sockaddr?

2024-10-21 09:14:00
admin
原创
115
摘要:问题描述:bind ()函数接受指向 的指针sockaddr,但在我见过的所有例子中,sockaddr_in都使用结构来代替,并转换为sockaddr:struct sockaddr_in name; ...; if (bind (sock, (struct sockaddr *) &name...

问题描述:

bind ()函数接受指向 的指针sockaddr,但在我见过的所有例子中,sockaddr_in都使用结构来代替,并转换为sockaddr

struct sockaddr_in name;
...;
if (bind (sock, (struct sockaddr *) &name, sizeof (name)) < 0) {
  ...;
}

我不明白为什么要sockaddr_in使用结构体。为什么不直接准备并传递一个sockaddr

这只是惯例吗?


解决方案 1:

不,这不仅仅是惯例。

sockaddr是任何类型的套接字操作的通用描述符,而是sockaddr_in基于 IP 通信的特定结构(如果我没记错的话,“in”代表“InterNet”)。据我所知,这是一种“多态性”:该bind()函数假装接受一个struct sockaddr *,但实际上,它会假定传入了适当类型的结构;即与您作为第一个参数提供的套接字类型相对应的结构。

解决方案 2:

我不知道这是否与这个问题非常相关,但我想提供一些额外的信息,这可能会使类型种姓更容易理解,因为许多没有花太多时间的人C看到这样的类型种姓会感到困惑。

我使用macOS,因此我根据系统中的头文件来举例。

struct sockaddr定义如下:

struct sockaddr {
    __uint8_t       sa_len;         /* total length */
    sa_family_t     sa_family;      /* [XSI] address family */
    char            sa_data[14];    /* [XSI] addr value (actually larger) */
};

struct sockaddr_in定义如下:

struct sockaddr_in {
    __uint8_t       sin_len;
    sa_family_t     sin_family;
    in_port_t       sin_port;
    struct  in_addr sin_addr;
    char            sin_zero[8];
};

从最基础的开始,指针只包含一个地址。所以struct sockaddr *struct sockaddr_in *几乎相同。它们都只存储一个地址。唯一相关的区别是编译器如何处理它们的对象。

因此,当您说时(struct sockaddr *) &name,您只是在欺骗编译器并告诉它这个地址指向一个struct sockaddr类型。


假设指针指向某个位置1000。如果struct sockaddr *存储了该地址,它将根据结构定义考虑从1000到的内存拥有成员。如果存储了相同的地址,它将考虑从到 的内存。sizeof(struct sockaddr)`struct sockaddr_in *1000sizeof(struct sockaddr_in)`


当您对该指针进行类型转换时,它将考虑相同的字节序列sizeof(struct sockaddr)

struct sockaddr *a = &name; // consider &name = 1000

现在,如果我访问a->sa_len,编译器将从 位置访问1000sizeof(__uint8_t)其字节大小与 的情况相同sockaddr_in。因此这应该访问相同的字节序列。

同样的模式也适用于sa_family

之后有一个 14 字节字符数组,其中struct sockaddr存储来自in_port_t sin_porttypedef16 位无符号整数 = 2 字节)、struct in_addr sin_addr(简单来说 32 位 ipv4 地址 = 4 字节)和char sin_zero[8](8 字节)的数据。这 3 个加起来是 14 个字节。

现在这三个都存储在这个 14 字节字符数组中,我们可以通过访问适当的索引并再次对其进行类型转换来访问这三个中的任何一个。

user529758 的回答已经解释了这样做的原因。

解决方案 3:

这是因为 bind 可以绑定除 IP 套接字之外的其他类型的套接字,例如 Unix 域套接字,其类型为 sockaddr_un。AF_INET 套接字的地址包含主机和端口,而 AF_UNIX 套接字包含文件系统路径。

相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用