完成到第三章3.4 P48

This commit is contained in:
riba2534
2019-01-13 14:11:31 +08:00
parent 40895a0ce5
commit 766d72b53f
4 changed files with 417 additions and 0 deletions

355
README.md
View File

@@ -457,7 +457,362 @@ Function read call count: 13
> TCP 不存在数据边界。在接收数据时需要保证在接收套接字的缓冲区填充满之时就从buffer里读取数据。也就是在接收套接字内部写入buffer的速度要小于读出buffer的速度。
## 第三章 地址族与数据序列
本章代码,在[TCP-IP-NetworkNote](https://github.com/riba2534/TCP-IP-NetworkNote)中可以找到。
把套接字比喻成电话,那么目前只安装了电话机,本章讲解给电话机分配号码的方法,即给套接字分配 IP 地址和端口号。
### 3.1 分配给套接字的 IP 地址与端口号
IP 是 Internet Protocol网络协议的简写是为手法网络数据而分配给计算机的值。端口号并非赋予计算机的值而是为了区分程序中创建的套接字而分配给套接字的端口号。
#### 3.1.1 网络地址Internet Address
为使计算机连接到网络并收发数据,必须为其分配 IP 地址。IP 地址分为两类。
- IPV4Internet Protocol version 44 字节地址族
- IPV6Internet Protocol version 66 字节地址族
两者之间的主要差别是 IP 地址所用的字节数,目前通用的是 IPV4 , IPV6 的普及还需要时间。
IPV4 标准的 4 字节 IP 地址分为网络地址和主机(指计算机)地址,且分为 A、B、C、D、E 等类型。
![](https://i.loli.net/2019/01/13/5c3ab0eb17bbe.png)
数据传输过程:
![](https://i.loli.net/2019/01/13/5c3ab19174fa4.png)
某主机向 203.211.172.103 和 203.211.217.202 传递数据,其中 203.211.172 和 203.211.217 为该网络的网络地址,所以「向相应网络传输数据」实际上是向构成网络的路由器或者交换机传输数据,然后又路由器或者交换机根据数据中的主机地址向目标主机传递数据。
#### 3.1.2 网络地址分类与主机地址边界
只需通过IP地址的第一个字节即可判断网络地址占用的总字节数因为我们根据IP地址的边界区分网络地址如下所示
- A 类地址的首字节范围为0~127
- B 类地址的首字节范围为128~191
- C 类地址的首字节范围为192~223
还有如下这种表示方式:
- A 类地址的首位以 0 开始
- B 类地址的前2位以 10 开始
- C 类地址的前3位以 110 开始
因此套接字手法数据时,数据传到网络后即可轻松找到主机。
#### 3.1.3 用于区分套接字的端口号
IP地址用于区分计算机只要有IP地址就能向目标主机传输数据但是只有这些还不够我们需要把信息传输给具体的应用程序。
所以计算机一般有 NIC网络接口卡数据传输设备。通过 NIC 接受的数据内有端口号,操作系统参考端口号把信息传给相应的应用程序。
端口号由 16 位构成,可分配的端口号范围是 0~65535 。但是 0~1023 是知名端口,一般分配给特定的应用程序,所以应当分配给此范围之外的值。
虽然端口号不能重复,但是 TCP 套接字和 UDP 套接字不会共用端接口号,所以允许重复。如果某 TCP 套接字使用了 9190 端口号,其他 TCP 套接字就无法使用该端口号,但是 UDP 套接字可以使用。
总之数据传输目标地址同时包含IP地址和端口号只有这样数据才会被传输到最终的目的应用程序。
### 3.2 地址信息的表示
应用程序中使用的IP地址和端口号以结构体的形式给出了定义。本节围绕结构体讨论目标地址的表示方法。
#### 3.2.1 表示 IPV4 地址的结构体
结构体的定义如下
```c
struct sockaddr_in
{
sa_family_t sin_family; //地址族Address Family
uint16_t sin_port; //16 位 TCP/UDP 端口号
struct in_addr sin_addr; //32位 IP 地址
char sin_zero[8]; //不使用
};
```
该结构体中提到的另一个结构体 in_addr 定义如下,它用来存放 32 位IP地址
```c
struct in_addr
{
in_addr_t s_addr; //32位IPV4地址
}
```
关于以上两个结构体的一些数据类型。
| 数据类型名称 | 数据类型说明 | 声明的头文件 |
| :----------: | :----------------------------------: | :----------: |
| int 8_t | signed 8-bit int | sys/types.h |
| uint8_t | unsigned 8-bit int (unsigned char) | sys/types.h |
| int16_t | signed 16-bit int | sys/types.h |
| uint16_t | unsigned 16-bit int (unsigned short) | sys/types.h |
| int32_t | signed 32-bit int | sys/types.h |
| uint32_t | unsigned 32-bit int (unsigned long) | sys/types.h |
| sa_family_t | 地址族address family | sys/socket.h |
| socklen_t | 长度length of struct | sys/socket.h |
| in_addr_t | IP地址声明为 uint_32_t | netinet/in.h |
| in_port_t | 端口号,声明为 uint_16_t | netinet/in.h |
为什么要额外定义这些数据类型呢?这是考虑扩展性的结果
#### 3.2.2 结构体 sockaddr_in 的成员分析
- 成员 sin_family
每种协议适用的地址族不同比如IPV4 使用 4 字节的地址族IPV6 使用 16 字节的地址族。
> 地址族
| 地址族Address Family | 含义 |
| ------------------------ | ---------------------------------- |
| AF_INET | IPV4用的地址族 |
| AF_INET6 | IPV6用的地址族 |
| AF_LOCAL | 本地通信中采用的 Unix 协议的地址族 |
AF_LOACL 只是为了说明具有多种地址族而添加的。
- 成员 sin_port
该成员保存 16 位端口号,重点在于,它以网络字节序保存。
- 成员 sin_addr
该成员保存 32 为IP地址信息且也以网络字节序保存
- 成员 sin_zero
无特殊含义。只是为结构体 sockaddr_in 结构体变量地址值将以如下方式传递给 bind 函数。
在之前的代码中
```c
if (bind(serv_sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) == -1)
error_handling("bind() error");
```
此处 bind 第二个参数期望得到的是 sockaddr 结构体变量的地址值包括地址族、端口号、IP地址等。
```c
struct sockaddr
{
sa_family_t sin_family; //地址族
char sa_data[14]; //地址信息
}
```
此结构体 sa_data 保存的地址信息中需要包含IP地址和端口号剩余部分应该填充 0 ,但是这样对于包含地址的信息非常麻烦,所以出现了 sockaddr_in 结构体,然后强制转换成 sockaddr 类型,则生成符合 bind 条件的参数。
### 3.3 网络字节序与地址变换
不同的 CPU 中4 字节整数值1在内存空间保存方式是不同的。
有些 CPU 这样保存:
```
00000000 00000000 00000000 00000001
```
有些 CPU 这样保存:
```
00000001 00000000 00000000 00000000
```
两种一种是顺序保存,一种是倒序保存 。
#### 3.3.1 字节序Order与网络字节序
CPU 保存数据的方式有两种,这意味着 CPU 解析数据的方式也有 2 种:
- 大端序Big Endian高位字节存放到低位地址
- 小端序Little Endian高位字节存放到高位地址
![big.png](https://i.loli.net/2019/01/13/5c3ac9c1b2550.png)
![small.png](https://i.loli.net/2019/01/13/5c3ac9c1c3348.png)
两台字节序不同的计算机在数据传递的过程中可能出现的问题:
![zijiexu.png](https://i.loli.net/2019/01/13/5c3aca956c8e9.png)
因为这种原因,所以在通过网络传输数据时必须约定统一的方式,这种约定被称为网络字节序,非常简单,统一为大端序。即,先把数据数组转化成大端序格式再进行网络传输。
#### 3.3.2 字节序转换
帮助转换字节序的函数:
```c
unsigned short htons(unsigned short);
unsigned short ntohs(unsigned short);
unsigned long htonl(unsigned long);
unsigned long ntohl(unsigned long);
```
通过函数名称掌握其功能,只需要了解:
- htons 的 h 代表主机host字节序。
- htons 的 n 代表网络network字节序。
- s 代表 short
- l 代表 long
下面的代码是示例,说明以上函数调用过程:
[endian_conv.c](https://github.com/riba2534/TCP-IP-NetworkNote/blob/master/ch03/endian_conv.c)
```cpp
#include <stdio.h>
#include <arpa/inet.h>
int main(int argc, char *argv[])
{
unsigned short host_port = 0x1234;
unsigned short net_port;
unsigned long host_addr = 0x12345678;
unsigned long net_addr;
net_port = htons(host_port); //转换为网络字节序
net_addr = htonl(host_addr);
printf("Host ordered port: %#x \n", host_port);
printf("Network ordered port: %#x \n", net_port);
printf("Host ordered address: %#lx \n", host_addr);
printf("Network ordered address: %#lx \n", net_addr);
return 0;
}
```
编译运行:
```shell
gcc endian_conv.c -o conv
./conv
```
结果:
```
Host ordered port: 0x1234
Network ordered port: 0x3412
Host ordered address: 0x12345678
Network ordered address: 0x78563412
```
这是在小端 CPU 的运行结果。大部分人会得到相同的结果,因为 Intel 和 AMD 的 CPU 都是小端序为标准。
### 3.4 网络地址的初始化与分配
#### 3.4.1 将字符串信息转换为网络字节序的整数型
sockaddr_in 中需要的是 32 位整数型,但是我们只熟悉点分十进制表示法,那么改如何把类似于 201.211.214.36 转换为 4 字节的整数类型数据呢 ?幸运的是,有一个函数可以帮助我们完成它。
```C
#include <arpa/inet.h>
in_addr_t inet_addr(const char *string);
```
具体示例:
[inet_addr.c](https://github.com/riba2534/TCP-IP-NetworkNote/blob/master/ch03/inet_addr.c)
```c
#include <stdio.h>
#include <arpa/inet.h>
int main(int argc, char *argv[])
{
char *addr1 = "1.2.3.4";
char *addr2 = "1.2.3.256";
unsigned long conv_addr = inet_addr(addr1);
if (conv_addr == INADDR_NONE)
printf("Error occured! \n");
else
printf("Network ordered integer addr: %#lx \n", conv_addr);
conv_addr = inet_addr(addr2);
if (conv_addr == INADDR_NONE)
printf("Error occured! \n");
else
printf("Network ordered integer addr: %#lx \n", conv_addr);
return 0;
}
```
编译运行:
```shell
gcc inet_addr.c -o addr
./addr
```
输出:
```
Network ordered integer addr: 0x4030201
Error occured!
```
1个字节能表示的最大整数是255所以代码中 addr2 是错误的IP地址。从运行结果看inet_addr 不仅可以转换地址,还可以检测有效性。
inet_aton 函数与 inet_addr 函数在功能上完全相同也是将字符串形式的IP地址转换成整数型的IP地址。只不过该函数用了 in_addr 结构体,且使用频率更高。
```c
#include <arpa/inet.h>
int inet_aton(const char *string, struct in_addr *addr);
/*
成功时返回 1 ,失败时返回 0
string: 含有需要转换的IP地址信息的字符串地址值
addr: 将保存转换结果的 in_addr 结构体变量的地址值
*/
```
函数调用示例:
[inet_aton.c](https://github.com/riba2534/TCP-IP-NetworkNote/blob/master/ch03/inet_aton.c)
```c
#include <stdio.h>
#include <stdlib.h>
#include <arpa/inet.h>
void error_handling(char *message);
int main(int argc, char *argv[])
{
char *addr = "127.232.124.79";
struct sockaddr_in addr_inet;
if (!inet_aton(addr, &addr_inet.sin_addr))
error_handling("Conversion error");
else
printf("Network ordered integer addr: %#x \n", addr_inet.sin_addr.s_addr);
return 0;
}
void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit(1);
}
```
编译运行:
```c
gcc inet_aton.c -o aton
./aton
```
运行结果:
```
Network ordered integer addr: 0x4f7ce87f
```
可以看出,已经成功的把转换后的地址放进了 addr_inet.sin_addr.s_addr 中。
还有一个函数,与 inet_aton() 正好相反。
## License

19
ch03/endian_conv.c Normal file
View File

@@ -0,0 +1,19 @@
#include <stdio.h>
#include <arpa/inet.h>
int main(int argc, char *argv[])
{
unsigned short host_port = 0x1234;
unsigned short net_port;
unsigned long host_addr = 0x12345678;
unsigned long net_addr;
net_port = htons(host_port); //转换为网络字节序
net_addr = htonl(host_addr);
printf("Host ordered port: %#x \n", host_port);
printf("Network ordered port: %#x \n", net_port);
printf("Host ordered address: %#lx \n", host_addr);
printf("Network ordered address: %#lx \n", net_addr);
return 0;
}

20
ch03/inet_addr.c Normal file
View File

@@ -0,0 +1,20 @@
#include <stdio.h>
#include <arpa/inet.h>
int main(int argc, char *argv[])
{
char *addr1 = "1.2.3.4";
char *addr2 = "1.2.3.256";
unsigned long conv_addr = inet_addr(addr1);
if (conv_addr == INADDR_NONE)
printf("Error occured! \n");
else
printf("Network ordered integer addr: %#lx \n", conv_addr);
conv_addr = inet_addr(addr2);
if (conv_addr == INADDR_NONE)
printf("Error occured! \n");
else
printf("Network ordered integer addr: %#lx \n", conv_addr);
return 0;
}

23
ch03/inet_aton.c Normal file
View File

@@ -0,0 +1,23 @@
#include <stdio.h>
#include <stdlib.h>
#include <arpa/inet.h>
void error_handling(char *message);
int main(int argc, char *argv[])
{
char *addr = "127.232.124.79";
struct sockaddr_in addr_inet;
if (!inet_aton(addr, &addr_inet.sin_addr))
error_handling("Conversion error");
else
printf("Network ordered integer addr: %#x \n", addr_inet.sin_addr.s_addr);
return 0;
}
void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit(1);
}