项目学习地址:【牛客网C++服务器项目学习】
day10
1.socket
socket是介于应用层和传输层的一种对外的接口。socket是网络间主机通信的API,它将传输层及其以下的网络层次都封装了,在应用层通过编写和调用socket函数即可实现对两个主机上进行通信。
2.字节序
节序,就是 多字节(大于一个字节)类型的数据在内存中的存放顺序。是在跨平台和网络编程中,时常要考虑的问题。
字节序经常被分为两类:
\1. Big-Endian(大端):高位字节排放在内存的低地址端,低位字节排放在内存的高地址端。
2.Little-Endian(小端):低位字节排放在内存的低地址端,高位字节排放在内存的高地址端。
给大家画个图就明白了:
设变量x为:int x = 0x01234567
,地址为0x100
字节序的转换函数:
//头文件:#include <arpa/inet.h>
//uint16_t:unsigned short int
//转换端口地址(端口地址长度为2个字节)
uint16_t ntohs(uint16_t netint16);
功能:
将16位网络字节序数据转换成主机字节序数据
参数:
netint16:待转换的16位网络字节序数据
返回值:
成功:返回主机字节序的值
uint16_t htons(uint16_t hostint16);
功能:
将16位主机字节序数据转换成网络字节序数据
参数:
hostint16:待转换的16位主机字节序数据
返回值:
成功:返回网络字节序的值
//转换IPv4地址
//uint32_t: unsigned int
uint32_t ntohl(uint32_t netint32);
功能:
将32位网络字节序数据转换成主机字节序数据
参数:
netint32:待转换的32位网络字节序数据
返回值:
成功:返回主机字节序的值
uint32_t htonl(uint32_t hostint32);
功能:
将32位主机字节序数据转换成网络字节序数据
参数:
hostint32:待转换的32位主机字节序数据
返回值:
成功:返回网络字节序的值
3.套接字地址结构
大多数套接字函数都需要一个指向套接字地址结构的指针作为参数。每一个协议族都定义了它自己的套接字地址结构。这些结构的名字均以sockaddr_来头,并以对应每个协议族位移后缀结尾
IPv4套接字地址结构
struct in_addr
{
in_addr_t s_addr; /* 32-bit IPv4 address */
}; /* network byet orered */
stuct sockaddr_in
{
uint8_t sin_len; /* length of structure */
sa_familyt sin_family; /* AF_INET */
in_port_t sin_port; /* 16-bit TCP or UDP port number */
/* network byet orered */
struct in_addr sin_addr; /* 32-bit IPv4 address */
/* network byet orered */
char sin_zero[8] /* unused */
};
-
长度字段sin_len是为了增加对OSI协议的支持添加的。在此之前,第一成员时sin_family,它是一个无符号短整数。并不是所有的厂家都支持套接字地址结构的长度字段。并且即使有长度字段,我们也无需设置和检查它,除非涉及路由套接字。
-
POSIX规范只需要这个结构中的3个字段: sin_family、sin_addr和sin_port。
-
IPv4地址和TCP或UDP端口号在套接字地址结构 中总是以网络字节序来存储。
-
32位IPv4地址存在两种不同的访问方法。举例来说。如果serv定义为某个网际套接字地址结构,那么serv.sin_addr将按in_addr结构引用其中的32位IPv4地址,而serv.sin_addr.s_addr将按in_addr_t(通常是一个无符号的32位整数)引用同一个32位IPv4地址。因此,我们必须正确地使用IPv4地址,尤其是在将它作为函数的参数时,因为编译器对传结构和传递整数的处理是完全不同的。
-
套接字地址结构仅在给定主机上使用:虽然结构的某些字段(例如IP地址和端口号)用在不同主机之间的通信中,但是结构本身并不在主机之间传递。
通用套接字地址结构
struct sockaddr
{
uint8_t sa_len;
sa_family_t sa_family; /* address family: AF_XXX value */
char sa_data[14]; /* protocol-specific address */
};
- 因为套接字函数出现得比ANSI C语言要早,后期为了规范程序开发,又设置了一个通用的sockaddr。其实这个通用的函数用得很少,更多地只是作为现在套接字函数参数的一个规范参数,我们仍旧可以使用之前的IPv4套接字函数,传递参数的时候加一个强制类型转换即可
IPv6套接字地址结构
struct in6_addr
{
uint8_t s6_addr[16]; /* 128-bit IPv6 address */
/* network byet orered */
};
#define SIN6_LEN /* required for compile-time tests */
stuct sockaddr_in6
{
uint8_t sin6_len; /* length of this structure (28)*/
sa_family_t sin6_family; /* AF_INET6 */
in_port_t sin6_port; /* transport layer port# */
/* network byet orered */
uint32_t sin6_flowinfo; /* flow information,undefined */
struct in6_addr sin6_addr; /* IPv6 address */
/* network byet orered */
uint32_t sin6_scope_id; /* set of interfaces for a scope */
};
4.IP地址转换函数
IP地址转换函数的功能,是将int类型和字符串类型进行一个相互转换
由于我们常见的IP地址都是用点分十进制表示的,在计算机中打印点分十进制一般都是使用字符串数组。然后如果我们要对IP地址进行校验或者其他处理,最好还是转换为32位的int整型。在linux中有这么几个函数可以直接调用
早期的三个转换函数
/* Convert Internet host address from numbers-and-dots notation in CP
into binary data and store the result in the structure INP. */
int inet_aton (const char *__cp, struct in_addr *__inp);
/* Convert Internet host address from numbers-and-dots notation in CP
into binary data in network byte order. */
in_addr_t inet_addr (const char *__cp);
/* Convert Internet number in IN to ASCII representation. The return value
is a pointer to an internal array containing the string. */
char *inet_ntoa (struct in_addr __in);
现在常用的两个函数
int inet_pton (int __af, const char *__restrict __cp,
void *__restrict __buf);
const char *inet_ntop (int __af, const void *__restrict __cp,
char *__restrict __buf, socklen_t __len);