前言:
学习视频:中科大郑烇、杨坚全套《计算机网络(自顶向下方法 第7版,James F.Kurose,Keith W.Ross)》课程
该视频是B站非常著名的计网学习视频,但相信很多朋友和我一样在听完前面的部分发现信息量过大,有太多无法理解的地方,在我第一次点开的时候也有相同的感受,但经过了一段时间项目的学习,对计网有了更多的了解,所以我准备在这次学习的时候做一些记录并且加入一些我的理解,希望能够帮助到大家。
往期笔记可以看专栏中的内容
文章目录
- 2.9 UDP 套接字编程
- 03. 传输层
- 3.1 概述和传输层的服务
- 3.1.1 传输服务和协议
- 3.1.2 传输层 VS 网络层
- 3.1.3 Internet 传输层协议
与 TCP 不同的是 UDP 在客户端和服务器之间没有连接
- 发送端在报文中要明确的制定目标的 IP 地址和端口号
- 服务器必须从收到的分组中提取出发送端的 IP 地址和端口号。
再来看一下 UDP 套接字编程的代码,与上节的 TCP 编程做一下对比
上节内容:中科大计网学习记录笔记(十二):TCP 套接字编程
因为很多具体的部分上节已经讲过了,这里直接上代码
服务端
// 创建 UDP 套接字
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0) {
perror("Socket creation failed");
exit(EXIT_FAILURE);
}
// 创建 TCP 套接字
if ((client_sock = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
perror("socket creation failed");
exit(EXIT_FAILURE);
}
可以看出创建 socket 时指定的套接字类型不同,分别为 SOCK_DGRAM
和 SOCK_STREAM
// 设置服务器地址结构
memset(&server_addr, 0, sizeof(server_addr));
memset(&client_addr, 0, sizeof(client_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = INADDR_ANY;
server_addr.sin_port = htons(PORT);
// 将套接字绑定到服务器地址
if (bind(sockfd, (const struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
perror("Bind failed");
exit(EXIT_FAILURE);
}
接下来是设置服务器的 sockaddr
并且将其与 socket 相关联
while (1) {
// 接收来自客户端的消息
n = recvfrom(sockfd, (char *)buffer, BUF_SIZE, 0, (struct sockaddr *)&client_addr, &len);
buffer[n] = '\0';
printf("Client : %s\n", buffer);
// 将接收到的消息发送回客户端
sendto(sockfd, (const char *)buffer, strlen(buffer), 0, (const struct sockaddr *)&client_addr, len);
}
在这里可以发现,服务器并没有 创建新的 socket 而是直接从输入中获得了客户端的信息来直接返回数据。
客户端
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0) {
perror("Socket creation failed");
exit(EXIT_FAILURE);
}
// 设置服务器地址结构
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(PORT);
server_addr.sin_addr.s_addr = inet_addr(SERVER_IP);
// 发送信息到服务器
sendto(sockfd, (const char *)buffer, strlen(buffer), 0, (const struct sockaddr *)&server_addr, sizeof(server_addr));
// 接收服务器的响应
n = recvfrom(sockfd, (char *)buffer, BUF_SIZE, 0, NULL, &len);
可以看出是直接发送的数据,而没有经过连接的过程
提纲:
3.1 概述和传输层服务
3.2 多路复用与解复用
3.3 无连接传输:UDP
3.4 可靠数据传输的原理
3.5 面向连接的传输 TCP
- 段结构
- 可靠数据传输
- 流量控制
- 连接管理
3.6 拥塞控制原理
3.7 TCP 拥塞控制
传输层为运行在 不同主机 上的 应用进程 提供了逻辑通信
- 在前面的学习中可以得知,网络层提供了端到端的通信,而传输层在网络层的基础上又提供了进程到进程的通信。
传输协议(如 TCP、UDP)运行在端系统上,发送方将应用层的报文分割成报文段(UDP 为数据报(datagram))
TCP 提供的是数据流的服务,而对于数据的分割需要应用层根据协议来判断
传输层提供进程之间的通信,网络层提供端到端的通信
在前面学到:本层对于上层的服务是本层的服务与其下面所有层提供的服务的总和
传输层作为网络层的上层在其端到端的通信的基础上做了如下的增强:
传输层对网络层做了几项主要的增强:
在这些里面需要注意的是多路复用:多路复用就是提供传输多个数据流来为多个进程提供服务,分为两步
来举一个现实中的例子:
Ann 家里的 12 个孩子给 Bill 家里的 12 个孩子发邮件
- Ann 家里将所有的邮件打包发出(复用)
- Bill 家里来讲邮件分发给不同的孩子(解复用)
类比:
- 主机 = 家庭
- 进程 = 小孩
- 应用层报文= 信封中的信件
- 传输协议= Ann 和 Bill
- 为家庭小孩提供复用解复用服 务
- 网络层协议 = 邮政服务
- 家庭-家庭的邮包传输服务
TCP
提供的服务:
UDP
提供的服务