2022-10-21-2022-10-23 网络编程

这篇博客展示了两个UDP协议的应用实例。第一个程序实现了客户端通过UDP从服务器下载文件,包括错误处理、数据包校验和确认机制。第二个程序演示了如何使用UDP进行广播通信,用户输入的消息会被发送到特定端口并广播到指定网络地址。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.

#include<stdio.h>
#include <sys/types.h>        
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include<string.h>
#include <sys/stat.h>
#include <fcntl.h>

#define ERR_MSG(msg) do{\
	fprintf(stderr,"line : %d",__LINE__);\
	perror(msg);\
	return -1;\
}while(0)

int main(int argc, const char *argv[])
{

	int cfd=socket( AF_INET , SOCK_DGRAM ,0);
	if(-1==cfd){
		ERR_MSG("socket errno");
	}

	struct sockaddr_in clientaddr;
	clientaddr.sin_family=AF_INET ;
	clientaddr.sin_port=htons(7777);
	clientaddr.sin_addr.s_addr=inet_addr("192.168.17.165");
	if(-1==bind(cfd,(struct sockaddr*)&clientaddr,sizeof(clientaddr))){
		ERR_MSG("bind errno");
	}
	printf("绑定客户端自身成功\n");

	// 填服务器的地址信息
	struct sockaddr_in serveraddr;
	serveraddr.sin_family=AF_INET ;
	serveraddr.sin_port=htons(69);
	serveraddr.sin_addr.s_addr=inet_addr("192.168.8.104");

	//发送
	char filename[32]="";
	int blocknum=0;      // 用于校验已经接到的数据包块编号
	struct sockaddr_in temp;
	socklen_t temp_len=sizeof(temp);
	int fd=0;
RE_NAME:   
	printf("请输入要下载的文件名>>>");
	scanf("%s",filename);

	// 组装请求包
	char buf[600]="";

	/*		// 组装方式1 读请求===>一个字节一个字节的赋值===>网络字节序
			buf[0]=0;
			buf[1]=1;
			*/

	/*		// 组装方式2 通过改变指针的操作空间来赋网络字节序的值
	 *(short *)buf=htons(1);
	 */
	// 组装方式3  使用格式字符串去赋值
	int nbytes=sprintf(buf,"%c%c%s%c%s%c",0,1,filename,0,"octet",0);

	// 发送请求包
	sendto(cfd,buf,nbytes,0,(struct sockaddr*)&serveraddr,sizeof(serveraddr));

	// 接受数据有可能是数据包有可能出错
	// 需要保存服务器的网络信息结构体因为临时端口号在里面
	// 定义变量存放临时端口
	short code=0;   // 用来保存操作码
	short num=0;    // 用来保存 要么是块编号 要么是差错码
	char txt[512]="";  // 保存要么是文件内容 要么是差错信息

	while(1){

		// 接收数据
		nbytes=recvfrom(cfd,buf,600,0,(struct sockaddr*)&temp,&temp_len);

		// 解析操作码

		code=ntohs(*(short*)buf);

		// 解析快编号或者是差错码
		num=ntohs(*(short*)(buf+2));

		// 解析文件内容或者差错信息
		strncpy(txt,buf+4,nbytes-4);

		if(code==5){
			printf("errrnocode=[%d]  errnotxt=[%s]\n",code,txt);
			goto RE_NAME;
		}else if(3==code&&blocknum+1=num){
			blocknum++;   // 用于校验下次接到的数据包
			if(1==num){  // 只有第一个数据包需要打开文件后面直接使用即可
				if(-1==(fd=open(filename,O_WRONLY|O_CREAT|O_TRUNC,0777))){
					ERR_MSG("open errno");
				}	
			}
			//将文件内容写入文件
			if(-1==write(fd,txt,nbytes-4)){
				ERR_MSG("write errno");
			}
			// 组装ACK
			*(short*)buf=htons(4);
			*(short*)(buf+2)=htons(num);
			
			// 发送ACK
			if(1-==sendto(cfd,buf,4,0,(struct sockaddr*)&temp,temp_len)){
				ERR_MSG("sendto errno"):
			}
			if(nbytes-4<512){
				break;
			}
		}
		close(fd);
		close(cfd);

		//		printf("code=[%d]  num=[%d]   txt=[%s]\n",code,num,txt);

	}



	return 0;
}

2.

#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netinet/ip.h>
#include<arpa/inet.h>
#include<string.h>
#include<unistd.h>

#define ERR_MSG(msg) do{\
	fprintf(stderr,"line:%d",__LINE__);\
	perror(msg);\
}while(0)


int main(int argc, const char *argv[])
{
	// 创建套接字	
	int cfd=socket(AF_INET,SOCK_DGRAM,0);
	if(cfd<0){
		ERR_MSG("socket");
		return -1;
	}
	printf("套接字创建成功\n");

	// 绑定发送端自身的IP和端口
	// 填充发送端的IP和端口
	struct sockaddr_in cin;
	cin.sin_family=AF_INET;
	cin.sin_port=htons(7777);
	cin.sin_addr.s_addr=inet_addr("192.168.8.102");
	
	if(bind(cfd,(struct sockaddr*)&cin,sizeof(cin))<0){
		ERR_MSG("bind");
		return -1;
	}
	printf("绑定成功\n");

	// 设置允许广播
	int set_broadcast=1;
	if(setsockopt(cfd,SOL_SOCKET,SO_BROADCAST,&set_broadcast,sizeof(set_broadcast))<0){
		ERR_MSG("set_broadcast");
		return -1;
	}

	// 填充接收端的IP和端口
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(6666);
	sin.sin_addr.s_addr=inet_addr("192.168.8.255");

	

	char buf[128]="";
	ssize_t res=0;

	// 接收数据
	while(1){
		
		bzero(buf,sizeof(buf));
		printf("请输入>>>");
		fgets(buf,sizeof(buf),stdin);
		buf[strlen(buf)-1]=0;

		if(sendto(cfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,sizeof(sin))<0){
			ERR_MSG("sendto");
			return -1;
		}
		printf("发送成功\n");
	}
	
	// 关闭文件描述符号
	close(cfd);

	return 0;
}

3.

#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netinet/ip.h>
#include<arpa/inet.h>
#include<string.h>
#include<unistd.h>

#define ERR_MSG(msg) do{\
	fprintf(stderr,"line:%d",__LINE__);\
	perror(msg);\
}while(0)


int main(int argc, const char *argv[])
{
	
	// 创建套接字
	int sfd=socket(AF_INET,SOCK_DGRAM,0);
	if(sfd<0){
		ERR_MSG("socket");
		return -1;
	}
	printf("套接字创建成功\n");


	// 填充接受端IP和端口
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(6666);
	sin.sin_addr.s_addr=inet_addr("192.168.8.255");

	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))<0){
		ERR_MSG("bind");
		return -1;
	}
	printf("绑定成功\n");

	struct sockaddr_in cin;       // 存储数据包是从哪里发送过来的
	socklen_t addrlen = sizeof(cin);
	
	char buf[128]="";
	ssize_t res=0;

	while(1){
		
		bzero(buf,sizeof(buf));

		res=recvfrom(sfd,buf,sizeof(buf),0,(struct sockaddr*)&cin,&addrlen);
		if(res<0){
			ERR_MSG("recvfrom");
			return -1;
		}

		printf("[%s:%d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
	}

	// 关闭文件描述符
	close(sfd);


	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值