UDP网络编程

本文详细介绍了UDP协议的特性,包括其无连接、不可靠传输的特点,并对比了与TCP协议的区别。阐述了UDP通信的基本流程,包括发送方和接收方的创建步骤,并提供了发送和接收数据的函数接口示例代码。此外,还展示了如何设置UDP套接字的广播特性以及广播发送方的示例。
摘要由CSDN通过智能技术生成

UDP网络编程

1.UDP协议简介

  UDP协议采用无连接的方式,不管发送的数据包是否到达目的主机,数据包是否出错。收到数据包的主机也不会告诉发送方是否正确收到了数据,它的可靠性是由上层协议来保障的。
  UDP 是User Datagram Protocol的简称, 中文名是用户数据报协议,是OSI(Open System Interconnection,开放式系统互联) 参考模型中一种无连接的传输层协议,提供面向事务的简单不可靠信息传送服务,IETF RFC 768 [1] 是UDP的正式规范。UDP在IP报文的协议号是17。
  UDP是无连接的服务。在无连接服务的情况下,两个实体之间的通信不需先建立好一个连接,因此其下层的有关资源不需要事先进行预定保留。这些资源将在数据传输时动态地进行分配。无连接服务的另一特征就是它不需要通信的两个实体同时是活跃的(即处于激活态)。当发送端的实体正在进行发送时,它才必须是活跃的。优点是灵活方便和比较迅速,但不能防止报文的丢失、重复或失序,特别适合于传送少量零星的报文。
  UDP报文没有可靠性保证、顺序保证和流量控制字段等,可靠性较差。但是正因为UDP协议的控制选项较少,在数据传输过程中延迟小、数据传输效率高,适合对可靠性要求不高的应用程序,或者可以保障可靠性的应用程序,如DNS、TFTP、SNMP等。
在这里插入图片描述
  UDP和TCP协议的主要区别是两者在如何实现信息的可靠传递方面不同。TCP协议中包含了专门的传递保证机制,当数据接收方收到发送方传来的信息时,会自动向发送方发出确认消息;发送方只有在接收到该确认消息之后才继续传送其它信息,否则将一直等待直到收到确认信息为止。与TCP不同,UDP协议并不提供数据传送的保证机制。如果在从发送方到接收方的传递过程中出现数据包的丢失,协议本身并不能做出任何检测或提示。因此,通常人们把UDP协议称为不可靠的传输协议。

2.UDP通讯流程

在这里插入图片描述
  一般在UDP通讯中我们不太区分服务端和客户端,由于UDP通讯不需要建立连接,因此UDP通讯中主要称为发送方和接收方。

  • 发送方创建过程:

  1.创建网络套接字socket
  2.发送数据sendto

  • 接收方创建过程:

  1.创建网络套接字socket
  2.绑定端口号
  3.接收数据recvfrom

2.1 函数接口

#include <sys/types.h>
#include <sys/socket.h>
发送数据
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);
形参: sockfd --套接字,socket函数返回值
   buf – 要发送是内容
   len --要发送的数据长度
   flags --一般填0即可
   dest_addr、addrlen —和connect后两个参数类似
   dest_addr —对方网络结构体信息
    addrlen --dest_addr结构体大小
返回值: 成功返回发送字节数,失败返回-1
接收数据
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);
形参: sockfd --套接字,socket函数返回值
    buf – 读取内容存放地址
    len --要读取的数据长度
    flags --一般填0即可
    src_addr、addrlen —和accept后两个参数类似
    src_addr —保存发送者的IP和端口号
    addrlen —src_addr结构体大小
返回值: 成功返回读取到的字节数,失败返回-1;

  • 发送方示例
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <stdlib.h>
#include <unistd.h>
#include <arpa/inet.h>
int main(int argc,char *argv[])
{
	if(argc!=3)
	{
		printf("格式:./a.out <端口号> <IP地址>\n");
		return 0;
	}
	int sockfd=socket(AF_INET,SOCK_DGRAM,0);	
	if(sockfd==-1)
	{
		printf("创建网络套接字失败\n");
		return 0;
	}
	struct sockaddr_in s_addr=
	{
		.sin_family=AF_INET,
		.sin_port=htons(atoi(argv[1])),
		.sin_addr.s_addr=inet_addr(argv[2]),//本地所有IP
	};
	char buff[]="UDP发送数据测试!";
	ssize_t size;
	while(1)
	{
		size=sendto(sockfd,buff,sizeof(buff),0,( const struct sockaddr * )&s_addr,sizeof(s_addr));
		printf("发送数据成功size=%ld\n",size);
		sleep(1);
	}
}
  • 接收方示例
#include <stdio.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc,char *argv[])
{
	if(argc!=2)
	{
		printf("./a.out <端口号>\n");
		return 0;
	}
	/*1.创建网络套接字*/
	int sockfd=socket(AF_INET,SOCK_DGRAM, 0);
	if(sockfd==-1)
	{
		printf("创建UDP网络套接字失败\n");
		return 0;
	}
	/*2.绑定端口号*/
	struct sockaddr_in addr=
	{
		.sin_family=AF_INET,
		.sin_port=htons(atoi(argv[1])),//发送的端口号
		.sin_addr.s_addr=INADDR_ANY,//本地所有IP
	};
	if(bind(sockfd,(const struct sockaddr *)&addr,sizeof(struct sockaddr)))
	{
		printf("绑定端口号失败\n");
		return 0;
	}
	/*开始接收数据*/
	char buff[256];
	struct sockaddr_in c_addr;
	socklen_t addrlen=sizeof(struct sockaddr_in);
	ssize_t size;
	while(1)
	{
		size=recvfrom(sockfd,buff,sizeof(buff)-1,0,(struct sockaddr *)&c_addr,&addrlen);
		if(size<=0)
		{
			printf("接收数据失败\n");
			continue;
		}
		buff[size]='\0';
		printf("[%s:%d] %s,len=%ld byte\n",inet_ntoa(c_addr.sin_addr),ntohs(c_addr.sin_port),buff,size);
	}
	close(sockfd);
}

在这里插入图片描述

2.2 设置UDP广播特性

  默认情况下UDP通讯是不支持广播特性,需要广播特性则需要设置UDP套接字属性。

//设置该套接字为广播类型,
	int nb = 0;
	nb = setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, (char *)&opt, sizeof(opt));
	if(nb == -1)
	{
		printf("设置广播类型错误.\n");
	}
  • 广播发送方示例
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <stdlib.h>
#include <unistd.h>
#include <arpa/inet.h>
int main(int argc,char *argv[])
{
	if(argc!=3)
	{
		printf("格式:./a.out <端口号> <IP地址>\n");
		return 0;
	}
	int sockfd=socket(AF_INET,SOCK_DGRAM,0);	
	if(sockfd==-1)
	{
		printf("创建网络套接字失败\n");
		return 0;
	}
	//设置该套接字为广播类型,
	const int opt = 1;
	int nb = 0;
	nb = setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, (char *)&opt, sizeof(opt));
	if(nb == -1)
	{
		printf("设置广播类型错误.\n");
	}
	struct sockaddr_in s_addr=
	{
		.sin_family=AF_INET,
		.sin_port=htons(atoi(argv[1])),
		.sin_addr.s_addr=inet_addr(argv[2]),//本地所有IP
	};
	char buff[]="UDP send data test,hello,world!";
	ssize_t size;
	while(1)
	{
		size=sendto(sockfd,buff,sizeof(buff),0,( const struct sockaddr * )&s_addr,sizeof(s_addr));
		printf("发送数据成功size=%ld\n",size);
		sleep(1);
	}
}

在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

IT_阿水

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值