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

2024-10-21 09:14:00
admin
原创
275
摘要:问题描述: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 套接字包含文件系统路径。

相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1120  
  IPD(Integrated Product Development,集成产品开发)流程是一种广泛应用于高科技和制造业的产品开发方法论。它通过跨职能团队的紧密协作,将产品开发周期缩短,同时提高产品质量和市场成功率。在IPD流程中,CDCP(Concept Decision Checkpoint,概念决策检查点)是一个关...
IPD培训课程   75  
  研发IPD(集成产品开发)流程作为一种系统化的产品开发方法,已经在许多行业中得到广泛应用。它不仅能够提升产品开发的效率和质量,还能够通过优化流程和资源分配,显著提高客户满意度。客户满意度是企业长期成功的关键因素之一,而IPD流程通过其独特的结构和机制,能够确保产品从概念到市场交付的每个环节都围绕客户需求展开。本文将深入...
IPD流程   66  
  IPD(Integrated Product Development,集成产品开发)流程是一种以跨职能团队协作为核心的产品开发方法,旨在通过优化资源分配、提高沟通效率以及减少返工,从而缩短项目周期并提升产品质量。随着企业对产品上市速度的要求越来越高,IPD流程的应用价值愈发凸显。通过整合产品开发过程中的各个环节,IPD...
IPD项目管理咨询   76  
  跨部门沟通是企业运营中不可或缺的一环,尤其在复杂的产品开发过程中,不同部门之间的协作效率直接影响项目的成败。集成产品开发(IPD)作为一种系统化的项目管理方法,旨在通过优化流程和增强团队协作来提升产品开发的效率和质量。然而,跨部门沟通的复杂性往往成为IPD实施中的一大挑战。部门之间的目标差异、信息不对称以及沟通渠道不畅...
IPD是什么意思   70  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

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

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

内置subversion和git源码管理

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

免费试用