文章目录
- IPV4套接字地址结构
- IPv6套接字地址结构
- 字节排序函数
- 地址转换函数
IPV4套接字地址结构
IPv4套接字定义在<netinet/in.h>
投文件中,定义如下:
struct in_addr {
in_addr_t s_addr;
}
struct sockaddr_in {
uint8_t sin_len; // 长度字段
sa_family_t sin_family; // 通常是一个8位无符号整数,在不支持长度字段实现中,是一个16位无符号整数
in_port_t sin_port; // 至少16位的无符号整数
struct in_addr sin_addr; // 32位IPv4地址
char sin_zero[8]; // 没用
};
可以看出IPv4地址访问方式有两种:一种是serv.sin_addr以结构体的方式使用,另一种及时访问进结构体,使用32位无符号整数。
作为参数传递套接字函数时,套接字地址以引用方式传递,需要强制转换成struct sockaddr*
类型,其在<sys/socket.h>
中定义为:
struct sockaddr {
uint8_t sa_len;
sa_family_t sa_family;
char sa_data[14]
};
IPv6套接字地址结构
struct in6_addr {
uint8_t s6_addr[16]; //128bit
};
#define SIN6_LEN
struct sockaddr_in6 {
uint8_t sin6_len;
sa_family_t sin6_family;
in_port_t sin6_port;
uint32_t sin6_flowinfo; //
struct in6_addr sin6_addr;
uint32_t sin6_scope_id; //表示具有范围的地址的范围
}
字节排序函数
判断系统是大端还是小端格式的方法
#include "stdio.h"
int main()
{
int a = 0x12345678;
char *p = (char *)&a;
printf("*p = %#x\n", *p);
if(*p == 0x78)
printf("小端\n");
else printf("大端\n");
return 0;
}
网络字节序是大端格式
两种字节序之间的转换可以用下面四个函数
// 将主机字节序转化为网络字节序
uint16_t htons(uint16_t host16bit_value);
uint32_t htonl(uint32_t host32bit_value);
// 将网络字节序,转化为主机字节序
uint16_t ntohs(uint16_t net16bit_value);
uint32_t ntohl(uint32_t net32bit_value);
尝试一下将主机字节序转换成网路字节序
地址转换函数
IPv4地址常使用点分十进制表示方法,在编程中要用二进制网络字节序,两者之间转换函数有:
inet_aton
, inet_addr
和inet_ntoa
,新的函数inet_pton
和inet_ntop
对于IPv4和IPv6都适用
#include <arpa/inet.h>
// 将字符串转换成二进制数形式
int inet_aton(const char *strptr, struct in_daar *addrptr)// 返回字符串有效1,无效0
in_addr_t inet_addr(const char *strptr) // 返回32位二进制网络字节序的IPv4地址
// 二进制转换成点分十进制形式
char *inet_ntoa(struct in_addr inaddr) // 返回指向点分十进制字符串的指针
inet_pton和inet_ntop函数
#include <arpa/inet.h>
// 两个函数的family可以是AF_INET, AF_INET6
// strptr所指的字符串转换成addptr所指的二进制数据
int inet_pton(int family, const char *strptr, void *addrptr);
// 从数值格式转换成点分十进制格式,len指示存放字符串的缓冲区的大小
const char *inet_ntop(int family, const void *addrptr, char *strptr, size_t len);
函数尝试
输出结果为: