UDP概念+编程流程
目录
[1.1 概念](#1.1 概念)
[1.2 特点](#1.2 特点)
[1.2.1 无连接性:](#1.2.1 无连接性:)
[1.2.2 不可靠性](#1.2.2 不可靠性)
[1.2.3 面向报文](#1.2.3 面向报文)
[2.1 客户端 cli.c](#2.1 客户端 cli.c)
[2.2 服务端ser.c](#2.2 服务端ser.c)
一、UDP基本概念
1.1 概念
UDP 即用户数据报协议(User Datagram Protocol ) ,是 OSI 参考模型中一种无连接的传输层协议,提供的是无连接,不可靠,数据报服务。
1.2 特点
1.2.1 无连接性:
UDP 传输数据之前源端和终端不建立连接,发送方直接将数据打包成 UDP 数据报发送到网络上,无需等待接收方确认,减少了开销和发送数据前的时延 。比如在线游戏中,玩家操作指令频繁发送,使用 UDP 可快速将指令发出,无需建立连接等待 。
1.2.2 不可靠性
UDP 不保证数据的可靠传输,不提供报文到达确认、排序及流量控制等功能 ,也没有数据包重传机制和错误恢复机制 。若传输中数据包丢失、损坏或乱序,UDP 不会自动处理 。像网络视频会议,偶尔丢包对整体影响不大,更看重实时性,就适合用 UDP 。
1.2.3 面向报文
UDP 对应用程序交下来的报文,在添加首部后就向下交付给 IP 层,既不拆分也不合并,保留报文边界 。这要求应用程序选择合适报文大小 。
二、UDP编程流程

2.1 客户端 cli.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
int main()
{
// 创建一个 UDP 套接字,AF_INET 表示使用 IPv4 地址族,SOCK_DGRAM 表示使用 UDP 协议
int sockfd = socket(AF_INET,SOCK_DGRAM,0);
// 检查套接字是否创建成功,如果创建失败(返回值为 -1),则终止程序
if( sockfd == -1 )
{
exit(1);
}
// 定义一个 IPv4 套接字地址结构体,用于存储服务器地址信息
struct sockaddr_in saddr;
// 将服务器地址结构体 saddr 的内存空间清零
memset(&saddr,0,sizeof(saddr));
// 设置地址族为 IPv4
saddr.sin_family = AF_INET;
// 将端口号 6000 从主机字节序转换为网络字节序
saddr.sin_port = htons(6000);
// 将 IP 地址 "127.0.0.1" 从点分十进制字符串转换为网络字节序的二进制数
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
// 进入无限循环,用于持续进行数据的发送和接收操作
while( 1 )
{
// 提示用户输入信息
printf("input:\n");
// 定义一个字符数组,用于存储用户输入的数据,大小为 128 字节
char buff[128] = {0};
// 从标准输入(键盘)读取一行数据到 buff 数组中,最多读取 128 个字符
fgets(buff,128,stdin);
// 检查用户输入的内容是否以 "end" 开头,如果是则跳出循环
if( strncmp(buff,"end",3) == 0)
{
break;
}
// 向服务器地址发送数据,strlen(buff) - 1 是去掉 fgets 读取时包含的换行符
sendto(sockfd,buff,strlen(buff)-1,0,(struct sockaddr*)&saddr,sizeof(saddr));
// 将字符数组 buff 的内容清零,为下次接收数据做准备
memset(buff,0,128);
// 初始化地址结构体长度变量,用于存储从服务器接收数据时地址结构体的长度
int len = sizeof(saddr);
// 从服务器接收数据,存储到 buff 数组中
recvfrom(sockfd,buff,127,0,(struct sockaddr*)&saddr,&len);
// 打印接收到的数据
printf("buff=%s\n",buff);
}
// 关闭套接字,释放相关资源
close(sockfd);
// 正常终止程序
exit(0);
}
sendto(sockfd,buff,strlen(buff)-1,0,(struct sockaddr*)&saddr,sizeof(saddr));
- 函数作用 :
sendto
是用于在 UDP 套接字上发送数据的函数。它可以将数据发送到指定的目标地址。- 参数解释
sockfd
:之前创建好的 UDP 套接字描述符,通过它来确定使用哪个套接字发送数据。buff
:这是一个字符数组,存储着要发送的数据内容,也就是用户通过fgets
从标准输入获取并准备发送给服务器的数据。strlen(buff) - 1
:strlen
函数用于获取buff
中字符串的长度。因为fgets
函数读取输入时会把换行符\n
也读取到buff
数组中,而实际发送数据时通常不需要这个换行符,所以减去 1 来去掉它 。这个值表示实际要发送的数据长度。0
:表示发送数据时的标志位,这里设置为 0 表示没有特殊的发送选项。(struct sockaddr*)&saddr
:将服务器地址结构体saddr
的指针强制转换为通用的sockaddr
指针类型,用于指定数据要发送到的目标地址,这里就是服务器的地址。sizeof(saddr)
:表示目标地址结构体saddr
的大小,用于告诉sendto
函数地址信息占用的字节数。recvfrom(sockfd,buff,127,0,(struct sockaddr*)&saddr,&len);
- 函数作用 :
recvfrom
是用于在 UDP 套接字上接收数据的函数,同时它还能获取发送方的地址信息。- 参数解释
sockfd
:UDP 套接字描述符,通过它确定从哪个套接字接收数据。buff
:用于存储接收到的数据的字符数组,这里它的大小为 128 ,但实际接收数据时预留 1 个字节用于存储字符串结束符\0
,所以指定接收长度为 127 。0
:接收数据时的标志位,设置为 0 表示没有特殊接收选项。(struct sockaddr*)&saddr
:用于存储发送方(客户端)地址信息的结构体指针,将其强制转换为通用的sockaddr
指针类型。在函数执行过程中,它会被填充为实际发送数据的客户端的地址信息。&len
:是一个指向表示地址结构体长度变量的指针。recvfrom
函数会根据实际接收到的地址信息长度来更新这个变量的值。
2.2 服务端ser.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
int main()
{
// 创建一个 UDP 套接字,AF_INET 表示使用 IPv4 地址族,SOCK_DGRAM 表示 UDP 协议
int sockfd = socket(AF_INET,SOCK_DGRAM,0);
// 检查套接字是否创建成功,若失败(返回 -1)则退出程序
if( -1 == sockfd)
{
exit(1);
}
// 定义两个 IPv4 套接字地址结构体,saddr 用于服务器地址,caddr 用于客户端地址
struct sockaddr_in saddr, caddr;
// 将服务器地址结构体 saddr 的内存初始化为 0
memset(&saddr,0,sizeof(saddr));
// 设置地址族为 IPv4
saddr.sin_family = AF_INET;
// 将端口号 6000 从主机字节序转换为网络字节序
saddr.sin_port = htons(6000);
// 将 IP 地址 "127.0.0.1" 从点分十进制转换为网络字节序
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
// 将套接字 sockfd 绑定到指定的地址(saddr)
int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
// 检查绑定操作是否成功,若失败(返回 -1)则打印错误信息
if( res == -1)
{
printf("bind err\n");
}
// 进入无限循环,持续接收和响应客户端消息
while( 1 )
{
// 定义字符数组 buff,用于存储接收到的消息,初始化为 0
char buff[128] = {0};
// 初始化客户端地址结构体 caddr 的长度
int len = sizeof(caddr);
// 从客户端接收数据,存储到 buff 中,同时获取客户端地址信息
int n = recvfrom(sockfd,buff,128,0,(struct sockaddr*)&caddr,&len);
// 打印接收到的消息内容
printf("buff=%s\n",buff);
// 向客户端发送 "ok" 消息
sendto(sockfd,"ok",2,0,(struct sockaddr*)&caddr,sizeof(caddr));
}
// 关闭套接字,释放资源
close(sockfd);
// 正常退出程序
exit(0);
}
int res = bind(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
这行代码的作用是将之前创建的套接字
sockfd
绑定到由saddr
结构体所表示的网络地址上,并通过返回值res
来判断绑定操作是否成功。
bind
函数是用于在网络编程中把一个套接字(socket)和一个特定的地址绑定在一起的系统函数。在 TCP 或 UDP 服务器程序中,通常使用bind
函数将服务器套接字绑定到一个指定的 IP 地址和端口号上,这样服务器就能在该地址上监听并接收来自客户端的连接请求或数据。