华清远见作业第二十九天——网络编程(第四天)

2024-01-17 23:12

本文主要是介绍华清远见作业第二十九天——网络编程(第四天),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

思维导图:

 基于UDP的TFTP文件传输

代码:

#include <a.h>
void menu();
int download(int cfd, struct sockaddr_in sin);
int up(int cfd, struct sockaddr_in sin);
#define SER_PORT 69
#define SER_IP "192.168.125.4"
int main(int argc, const char *argv[])
{//1创建用于通信的套接字文件描述符int cfd=socket(AF_INET,SOCK_DGRAM,0);if(cfd==-1){perror("111socket error:");return -1;}//2绑定//3填充服务器的地址信息结构体struct sockaddr_in sin;sin.sin_family=AF_INET;sin.sin_port=htons(SER_PORT);sin.sin_addr.s_addr=inet_addr(SER_IP);//选择语句int choose;while(1){menu();printf("请输入相关操作:");scanf("%d",&choose);getchar();switch(choose){case 1:{//下载功能download(cfd,sin);break;}case 2:{//上传功能up(cfd,sin);break;}case 3:{//退出功能//download(cfd,sin);goto END;break;}}}
END: close(cfd);return 0;}
//下载
int download(int cfd, struct sockaddr_in sin)
{char downloadname[128]="";printf("请输入你要下载的文件:");fgets(downloadname,120,stdin);downloadname[strlen(downloadname)-1]=0;//向服务器发送下载请求//向服务器发送下载请求char buf[512+2+2]=""; //数据包//组装请求数据short *p1=(short *)buf;*p1=htons(1);char *p2=buf+2;strcpy(p2,downloadname);  // 文件名字char *p3=p2+strlen(p2)+1;strcpy(p3,"octet");  //文件传输模式int len=4+strlen(p2)+strlen(p3); //要发送的长度//向服务器发送请求sendto(cfd,buf,len,0,(struct sockaddr*)&sin,sizeof(sin));//循环接收发送应答包ssize_t recvlen; //存放客户端发来消息函数的返回值(读取数据的个数)int fd=-1;//文件描述符号unsigned short num=1; //每一块的数据编号,初始值为1 //为啥是无符号型的我也不知道,有符号的就是错的那个机械臂也是必须要无符号的,真是服了socklen_t addrlen = sizeof(sin);int flag=0;  //防止被写入的文件重复打开while(1){	//清空数据包bzero(buf,sizeof(buf));//读取服务器发回来的信息(读取数据的个数)recvlen=recvfrom(cfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,&addrlen);if(recvlen==-1){printf("发送失败\n");perror("发送error:");return -1;}if(3==buf[1])   //前两位是3为数据包{	if(0 == flag)     //防止文件重复打开{//创建并打开文件,准备存储fd = open(downloadname, O_WRONLY|O_CREAT|O_TRUNC, 0664);  //文件名字if(fd==-1){perror("open error");return -1;}flag=1;}//判断数据包编号是否是自己想要的if(htons(num)==*(unsigned short*)(buf+2))  //强转为short并取值与num的网络字节序比{//判断成功后开始从数据包中提取数据写入到刚刚创建的文件中int geshu=write(fd,buf+4,recvlen-4);//减去前面的操作码和块编号if(geshu==-1){printf("写入错误\n");perror("写入错误:");return -1;}//回复ack包//ack的全部内容和数据包的前四位内容一样//修改操作码buf[1]=4;//向服务器发送ackgeshu=sendto(cfd,buf,4,0,(struct sockaddr*)&sin,sizeof(sin));if(geshu==-1){printf("ack发送错误\n");return -1;}//判断是否传输结束if(recvlen<512+2+2) //数据包要加上操作位和标志位{printf("文件传输完成了\n");break;}num++;  //每一个块编号加1}}else if(5==buf[1]) // 前两位是5表示内容错误,错误包{//错误printf("错误信息为:%s\n",buf+4);break;}}return 0;
}
//上传功能
int up(int cfd, struct sockaddr_in sin)
{char upname[128]="";printf("请输入你要上传的文件:");fgets(upname,120,stdin);upname[strlen(upname)-1]=0;//判断文件是否存在以只读的形式打开int fd=open(upname,O_RDONLY);if(fd==-1){printf("文件不存在,或者打开错误\n");return -1;}printf("打开成功\n");//核心上传给服务器char buf[512+2+2]=""; //数据包//组装请求数据short *p1=(short *)buf;*p1=htons(2); //请求写入char *p2=buf+2;strcpy(p2,upname);  // 文件名字char *p3=p2+strlen(p2)+1;strcpy(p3,"octet");  //文件传输模式int len=4+strlen(p2)+strlen(p3); //要发送的长度//向服务器发送请求sendto(cfd,buf,len,0,(struct sockaddr*)&sin,sizeof(sin));//上传的模块客户端循环向服务器发送数据ssize_t recvlen; //存放从服务器的返回值(读取数据的个数)unsigned short num=0; //每一块的数据编号,初始值为0 socklen_t addrlen = sizeof(sin);//循环向服务器发送信息	while(1){//清空数据包bzero(buf,sizeof(buf));//读取服务器发回来的信息(读取数据的个数)recvlen=recvfrom(cfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,&addrlen);if(recvlen==-1){printf("发送失败\n");perror("发送error:");return -1;}if(4==buf[1])	//第一次先返回ack{printf("1111111111\n");//判断块编号if(htons(num)==*(unsigned short*)(buf+2)){buf[1]=3;//如果是则块编号加1num++;*(unsigned short*)(buf+2)=htons(num);//在转换为主机字节排序//从客户端上面开始读取文件,通read函数读取fd,存到buf的数据位//du_num来统计读取的个数,如果读取的时候du_num=0则代表读取完成了int du_num;du_num=read(fd,buf+4,512);if(du_num==-1){perror("读取错误:");return -1;}else if(du_num==0){printf("上传完成了\n");break;}//向服务器发送数据包sendto(cfd,buf,du_num+4,0,(struct sockaddr*)&sin,sizeof(sin));printf("2222\n");}else{printf("错误了\n");break;}}else if(5==buf[1]) // 前两位是5表示内容错误,错误包{//错误printf("错误信息为:%s\n",buf+4);break;}		}return 0;
}//目录
void menu()
{printf("******************\n");printf("*     1.下载     *\n");printf("*     2.上传     *\n");printf("*     3.退出     *\n");printf("******************\n");	
}

运行效果:

 

这篇关于华清远见作业第二十九天——网络编程(第四天)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/617472

相关文章

作业提交过程之HDFSMapReduce

作业提交全过程详解 (1)作业提交 第1步:Client调用job.waitForCompletion方法,向整个集群提交MapReduce作业。 第2步:Client向RM申请一个作业id。 第3步:RM给Client返回该job资源的提交路径和作业id。 第4步:Client提交jar包、切片信息和配置文件到指定的资源提交路径。 第5步:Client提交完资源后,向RM申请运行MrAp

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

ASIO网络调试助手之一:简介

多年前,写过几篇《Boost.Asio C++网络编程》的学习文章,一直没机会实践。最近项目中用到了Asio,于是抽空写了个网络调试助手。 开发环境: Win10 Qt5.12.6 + Asio(standalone) + spdlog 支持协议: UDP + TCP Client + TCP Server 独立的Asio(http://www.think-async.com)只包含了头文件,不依

poj 3181 网络流,建图。

题意: 农夫约翰为他的牛准备了F种食物和D种饮料。 每头牛都有各自喜欢的食物和饮料,而每种食物和饮料都只能分配给一头牛。 问最多能有多少头牛可以同时得到喜欢的食物和饮料。 解析: 由于要同时得到喜欢的食物和饮料,所以网络流建图的时候要把牛拆点了。 如下建图: s -> 食物 -> 牛1 -> 牛2 -> 饮料 -> t 所以分配一下点: s  =  0, 牛1= 1~

poj 3068 有流量限制的最小费用网络流

题意: m条有向边连接了n个仓库,每条边都有一定费用。 将两种危险品从0运到n-1,除了起点和终点外,危险品不能放在一起,也不能走相同的路径。 求最小的费用是多少。 解析: 抽象出一个源点s一个汇点t,源点与0相连,费用为0,容量为2。 汇点与n - 1相连,费用为0,容量为2。 每条边之间也相连,费用为每条边的费用,容量为1。 建图完毕之后,求一条流量为2的最小费用流就行了

poj 2112 网络流+二分

题意: k台挤奶机,c头牛,每台挤奶机可以挤m头牛。 现在给出每只牛到挤奶机的距离矩阵,求最小化牛的最大路程。 解析: 最大值最小化,最小值最大化,用二分来做。 先求出两点之间的最短距离。 然后二分匹配牛到挤奶机的最大路程,匹配中的判断是在这个最大路程下,是否牛的数量达到c只。 如何求牛的数量呢,用网络流来做。 从源点到牛引一条容量为1的边,然后挤奶机到汇点引一条容量为m的边

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

深入理解RxJava:响应式编程的现代方式

在当今的软件开发世界中,异步编程和事件驱动的架构变得越来越重要。RxJava,作为响应式编程(Reactive Programming)的一个流行库,为Java和Android开发者提供了一种强大的方式来处理异步任务和事件流。本文将深入探讨RxJava的核心概念、优势以及如何在实际项目中应用它。 文章目录 💯 什么是RxJava?💯 响应式编程的优势💯 RxJava的核心概念