struct sockaddr_in, struct sockaddr,struct in_addr

一、结构体 struct sockaddr_in,  struct sockaddr,  struct in_addr

struct sockaddr_in,  struct sockaddr,struct in_addr,这是网络编程中常用的结构体,每次都记不住它们各自的成员是啥,需要临时查,为方便以后的查看,在这里总结下。

struct sockaddr {
unsigned short sa_family;  /* 地址族, AF_xxx */
char sa_data[14];  /* 14字节的协议地址*/
};

上面是通用的socket地址,具体到Internet socket,用下面的结构,二者可以进行类型转换
struct sockaddr_in {
short int sin_family; /* 地址族 */
unsigned short int sin_port; /* 端口号 */
struct in_addr sin_addr; /* Internet地址 */
unsigned char sin_zero[8]; /* 与struct sockaddr一样的长度 */
};

struct in_addr就是32位IP地址。
struct in_addr {
unsigned long s_addr;
};

也有
struct in_addr {
union {
struct { u_char s_b1,s_b2,s_b3,s_b4;} S_un_b;
struct { u_short s_w1,s_w2;} S_un_w;
u_long S_addr;
} S_un;
};
利用u_long htonl(u_long hostlong);将主机字节序转换为TCP/IP网络字节序.
利用u_short htons(u_short hostshort);将主机字节序转换为TCP/IP网络字节序.

inet_addr()是将一个点分制的IP地址(如192.168.0.1)转换为上述结构中需要的32位IP地址(0xC0A80001)。

通常的用法是:
int sockfd;
struct sockaddr_in my_addr;
sockfd = socket(AF_INET, SOCK_STREAM, 0); /* 做一些错误检查! */

my_addr.sin_family = AF_INET; /* 主机字节序 */
my_addr.sin_port = htons(MYPORT); /* short, 网络字节序 */
my_addr.sin_addr.s_addr = inet_addr(“192.168.0.1″);

bzero(&(my_addr.sin_zero), 8); /* zero the rest of the struct */
/* 不要忘了为bind()做错误检查: */
bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));

二、inet_addr、 inet_aton、inet_ntoa

inet_addr:  将网络地址转为网络二进制数字,返回的IP地址是网络序的。函数原型: unsigned long in inet_addr(const char *cp)

inet_aton:将网络地址转为网络二进制数字,与inet_addr的区别是,结果不是作为返回值,而是保存形参inp所指的in_addr结构体中。函数原型:int inet_aton(cont char* cp, struct in_addr *inp)

inet_ntoa:将网络二进制数字转为网络地址,函数原型是: char *inet_ntoa(struct in_addr in)

、 有两个更新的函数inet_pton和inet_ntop,这2个函数能够处理ipv4和ipv6,原型如下

int inet_pton(int af, const char *src, void *dst);
这个函数转换字符串到网络地址,第一个参数af是地址族,转换后存在dst中

inet_pton 是inet_addr的扩展,支持的多地址族有下列:

AF_INET:src为指向字符型的地址,即ASCII的地址的首地址(ddd.ddd.ddd.ddd格式的),函数将该地址转换为in_addr的结构体,并复制在*dst中

AF_INET6:rc为指向IPV6的地址,,函数将该地址转换为in6_addr的结构体,并复制在*dst中

如果函数出错将返回一个负值,并将errno设置为EAFNOSUPPORT,如果参数af指定的地址族和src格式不对,函数将返回0。

 

函数inet_ntop进行相反的转换原型如下
const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt);
这个函数转换网络二进制结构到ASCII类型的地址,参数的作用和上面相同,只是多了一个参数socklen_t cnt,他是所指向缓存区dst的大小,避免溢出,如果缓存区太小无法存储地址的值,则返回一个空指针,并将errno置为ENOSPC

`sockaddr_in` 是一种用于网络编程中的结构体,定义在 Unix/Linux 系统的头文件 `<netinet/in.h>` 中。它用于表示一个 Internet 地址。这个结构体通常与套接字编程接口一起使用,尤其是在设置或获取 IP 地址和端口号时。 下面是 `sockaddr_in` 结构体的一个标准定义(不同平台的定义可能略有不同): ```c struct sockaddr_in { sa_family_t sin_family; // 地址族 AF_INET 表示 IPv4 in_port_t sin_port; // 端口号,使用网络字节序 struct in_addr sin_addr; // IP 地址,使用网络字节序 // 通常会有一个填充字段保持结构体大小为 16 字节 unsigned char __pad[X]; // 在某些实现中,可能不需要填充,或者填充大小不同 }; ``` 在使用 `sockaddr_in` 结构体之前,需要对其进行初始化。比如,你可以使用 `inet_addr()` 函数将字符串形式的 IP 地址转换为网络字节序的格式,并使用 `htons()` 函数将端口号转换为网络字节序。 下面是一个初始化 `sockaddr_in` 结构体的例子: ```c #include <netinet/in.h> #include <arpa/inet.h> #include <sys/socket.h> struct sockaddr_in saddr; memset(&saddr, 0, sizeof(saddr)); // 清零内存 saddr.sin_family = AF_INET; // IPv4 saddr.sin_port = htons(8080); // 端口号,假设为8080 inet_pton(AF_INET, "192.168.1.1", &saddr.sin_addr); // IP地址,假设为192.168.1.1 ``` 这段代码将创建一个 `sockaddr_in` 结构体实例 `saddr`,并用 IP 地址 "192.168.1.1" 和端口号 8080 进行初始化,准备用于套接字编程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值