9.12 TFTP通信

客户端设计(仅供参考):

下载本质:读取服务器发送的数据包,写入到本地文件
上传本质:读取本地文件内容,发送给服务器。
1、建立菜单选项,上传和下载。
2、上传功能函数:
1、只读打开本地文件
2、发送上传请求数据包
3、接收服务器应答并回复ACK包
3、1如果接收的是ACK确认包,将包修改为数据包填充本地文件内容并发送给服务器
3、2如果是错误包,上传结束,检查网络。
3、下载功能函数
1、发送下载请求包
2、循环接收服务器数据包
2.1判断收到的包如果是数据包并且块编号正确,就写入客户端文件,然后回复ACK确认包
2.2如果包的长度小于512+2+2,说明接收完毕
2.3如果是错误包。输出对应错误信息并停止接收。

TFTP下载模型:

在这里插入图片描述

TFTP上传模型

在这里插入图片描述

TFTP通信过程总结

1、服务器在69号端口等待客户端的请求
2、服务器若批准此请求,则使用临时端口与客户端进行通信
3、每个数据包的编号都有变化(从1开始)
4、每个数据包都要得到ACK的确认如果出现超时,则需要重新发送最后的包(数据或
ACK)
5、数据的长度以512Byte传输
6、小于512Byte的数据意味着传输结束

do_upload(上传文件)

void do_upload(int sockfd,struct sockaddr_in sin,const char *filename){
	char text[1024] = "";
	int text_len;
	socklen_t sinlen = sizeof(sin);
	int fd;
	int flags = 0;
	int num = 0;
	ssize_t bytes;
	text_len = sprintf(text,"%c%c%s%c%s%c",0,2,filename,0,"octet",0);
	if(sendto(sockfd,text,text_len,0,(struct sockaddr *)&sin,sinlen)<0){
		perror("sendto");
		exit(1);
	}
	while(1){
		if((bytes = recvfrom(sockfd,text,sizeof(text),0,(struct sockaddr *)&sin,&sinlen))<0){
			perror("recvfrom");
			exit(1);
		}
		printf("接收操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));

		if(text[1] == 5){
			printf("差错码是:%d,差错信息error:%s\n",ntohs(*(short *)(text+2)),text+4);
			return ;
		}
		else if(text[1] == 4){
			if(flags == 0){
				if((fd = open(filename,O_RDONLY))<0){
					perror("open");
					exit(1);
				}
				flags = 1;
				printf("文件打开成功:%d\n",fd);
			}
			int len = read(fd,text+4,512);
			if (len < 0) { //检查 read 函数的返回值
                perror("read");
                exit(1);
            }
		//	printf("%s\n",text+4);
			if(len == 0){
				break;
			}
			text[1] = 3;
			*(short *)(text+2) = htons(++num);
			printf("发送操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));
			if(sendto(sockfd,text,len+4,0,(struct sockaddr *)&sin,sinlen)<0){
				perror("sendto");
				exit(1);
			}
		}
	}
}

do_download(下载文件)

void do_download(int sockfd,struct sockaddr_in sin,const char *filename){
	char text[1024] = "";
	int text_len;
	socklen_t sinlen = sizeof(sin);
	int fd;
	int flags = 0;
	int num = 0;
	ssize_t bytes;
	
	text_len = sprintf(text,"%c%c%s%c%s%c",0,1,filename,0,"octet",0);
	if(sendto(sockfd,text,text_len,0,(struct sockaddr *)&sin,sinlen)<0){
		perror("sendto");
		exit(1);
	}
	while(1){
		if((bytes = recvfrom(sockfd,text,sizeof(text),0,(struct sockaddr *)&sin,&sinlen))<0){
			perror("recvfrom");
			exit(1);
		}
		printf("操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));

		if(text[1] == 5){
			printf("错误码是:%d,错误信息error:%s\n",ntohs(*(short *)(text+2)),text+4);
			return;
		}
		else if(text[1]==3){
			if(flags==0){
				if((fd = open(filename,O_WRONLY|O_CREAT|O_TRUNC,0664))<0){
					perror("open");
					exit(1);
				}
				flags = 1;
			}
			if((num+1 == ntohs(*(short *)(text+2))) && (bytes == 516)){
				num = ntohs(*(short *)(text+2));
				if(write(fd,text+4,bytes-4)<0){
					perror("write");
					exit(1);
				}
				text[1] = 4;
				if(sendto(sockfd,text,4,0,(struct sockaddr *)&sin,sinlen)<0){
					perror("sendto");
					exit(1);
				}
			}
			else if((num+1 == ntohs(*(short *)(text+2)))&&(bytes<516)){
				if(write(fd,text+4,bytes-4)<0){
					perror("write");
					exit(1);
				}
				text[1]=4;
				if(sendto(sockfd,text,4,0,(struct sockaddr *)&sin,sinlen)<0){
					perror("sendto");
					exit(1);
				}
				printf("文件下载完毕\n");
				return;
			}
		}
	}
	close(fd);
}

运行结果

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

完整代码

#include <myhead.h>
#define SERPORT 69
#define SERIP "192.168.11.1"

void do_download(int sockfd,struct sockaddr_in sin,const char *filename);
void do_upload(int sockfd,struct sockaddr_in sin,const char *filename);

int main(int argc, const char *argv[])
{
    //1.创建用于通信的套接字文件描述符
    int clienfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (clienfd == -1) {
        perror("socket");
        return -1;
    }
    //2.填充服务器的地址信息结构体
    struct sockaddr_in sin = {
        .sin_family = AF_INET,
        .sin_port = htons(SERPORT),
        .sin_addr.s_addr = inet_addr(SERIP)
    };
    //3.菜单栏选择
    int choice;
    char filename[100];
    while (1) { //3.菜单栏
        printf("请选择操作:\n");
        printf("1.上传文件\n");
        printf("2.下载文件\n");
        printf("3.退出\n");
        scanf("%d", &choice);
        bzero(filename, sizeof(filename));
        if (choice == 1) {
            printf("请输入要上传的文件名:");
            scanf("%s", filename);
            do_upload(clienfd, sin, filename);
        } else if (choice == 2) {
            printf("请输入要下载的文件名:");
            scanf("%s", filename);
            do_download(clienfd, sin, filename);
        } else if (choice == 3) {
            break;
        } else {
            printf("无效选项,请重新选择。\n");
        }
    }
    close(clienfd);
    return 0;
}

void do_download(int sockfd,struct sockaddr_in sin,const char *filename){
	char text[1024] = "";
	int text_len;
	socklen_t sinlen = sizeof(sin);
	int fd;
	int flags = 0;
	int num = 0;
	ssize_t bytes;
	
	text_len = sprintf(text,"%c%c%s%c%s%c",0,1,filename,0,"octet",0);
	if(sendto(sockfd,text,text_len,0,(struct sockaddr *)&sin,sinlen)<0){
		perror("sendto");
		exit(1);
	}
	while(1){
		if((bytes = recvfrom(sockfd,text,sizeof(text),0,(struct sockaddr *)&sin,&sinlen))<0){
			perror("recvfrom");
			exit(1);
		}
		printf("操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));

		if(text[1] == 5){
			printf("错误码是:%d,错误信息error:%s\n",ntohs(*(short *)(text+2)),text+4);
			return;
		}
		else if(text[1]==3){
			if(flags==0){
				if((fd = open(filename,O_WRONLY|O_CREAT|O_TRUNC,0664))<0){
					perror("open");
					exit(1);
				}
				flags = 1;
			}
			if((num+1 == ntohs(*(short *)(text+2))) && (bytes == 516)){
				num = ntohs(*(short *)(text+2));
				if(write(fd,text+4,bytes-4)<0){
					perror("write");
					exit(1);
				}
				text[1] = 4;
				if(sendto(sockfd,text,4,0,(struct sockaddr *)&sin,sinlen)<0){
					perror("sendto");
					exit(1);
				}
			}
			else if((num+1 == ntohs(*(short *)(text+2)))&&(bytes<516)){
				if(write(fd,text+4,bytes-4)<0){
					perror("write");
					exit(1);
				}
				text[1]=4;
				if(sendto(sockfd,text,4,0,(struct sockaddr *)&sin,sinlen)<0){
					perror("sendto");
					exit(1);
				}
				printf("文件下载完毕\n");
				return;
			}
		}
	}
	close(fd);
}

void do_upload(int sockfd,struct sockaddr_in sin,const char *filename){
	char text[1024] = "";
	int text_len;
	socklen_t sinlen = sizeof(sin);
	int fd;
	int flags = 0;
	int num = 0;
	ssize_t bytes;
	text_len = sprintf(text,"%c%c%s%c%s%c",0,2,filename,0,"octet",0);
	if(sendto(sockfd,text,text_len,0,(struct sockaddr *)&sin,sinlen)<0){
		perror("sendto");
		exit(1);
	}
	while(1){
		if((bytes = recvfrom(sockfd,text,sizeof(text),0,(struct sockaddr *)&sin,&sinlen))<0){
			perror("recvfrom");
			exit(1);
		}
		printf("接收操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));

		if(text[1] == 5){
			printf("差错码是:%d,差错信息error:%s\n",ntohs(*(short *)(text+2)),text+4);
			return ;
		}
		else if(text[1] == 4){
			if(flags == 0){
				if((fd = open(filename,O_RDONLY))<0){
					perror("open");
					exit(1);
				}
				flags = 1;
				printf("文件打开成功:%d\n",fd);
			}
			int len = read(fd,text+4,512);
			if (len < 0) { //检查 read 函数的返回值
                perror("read");
                exit(1);
            }
		//	printf("%s\n",text+4);
			if(len == 0){
				break;
			}
			text[1] = 3;
			*(short *)(text+2) = htons(++num);
			printf("发送操作码:%d,块编号:%hd\n",text[1],ntohs(*(short *)(text+2)));
			if(sendto(sockfd,text,len+4,0,(struct sockaddr *)&sin,sinlen)<0){
				perror("sendto");
				exit(1);
			}
		}
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/875589.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【程序分享】Warren Cowley Parameters 程序:表征短程有序的化学基序

分享一个 Warren Cowley Parameters 程序&#xff1a;表征短程有序的化学基序。 感谢论文的原作者&#xff01; 主要内容 “晶体材料的化学成分具有原子尺度的波动&#xff0c;可调节各种中尺度特性。建立此类材料的化学-微结构关系需要对这些化学波动进行适当的表征。然而&…

2024网安周今日开幕,亚信安全亮相30城

2024年国家网络安全宣传周今天在广州拉开帷幕。今年网安周继续以“网络安全为人民&#xff0c;网络安全靠人民”为主题。2024年国家网络安全宣传周涵盖了1场开幕式、1场高峰论坛、5个重要活动、15场分论坛/座谈会/闭门会、6个主题日活动和网络安全“六进”活动。亚信安全出席20…

yolov8-obb中存在的一个bug

yolov8支持OBB目标检测,且能提供较好的性能。 但是最近在使用yolov8-obb的过程中,发现yolov8-obb存在一个bug。即训练数据如果包含不带旋转角度的水平目标时,训练出的模型,经常会输出垂直的检测框,需要旋转90度以后才能得到最终结果。把yolov8-obb相关的源码阅读一遍才发…

【数学建模】2024数学建模国赛经验分享

文章目录 一、关于我二、我的数模历程三、经验总结&#xff1a; 一、关于我 我的CSDN主页&#xff1a;https://gxdxyl.blog.csdn.net/ 2020年7月&#xff08;大二结束的暑假&#xff09;开始在CSDN写作&#xff1a; 阿里云博客专家&#xff1a; 接触的领域挺多的&#xff…

HTML 转 PDF API 接口

HTML 转 PDF API 接口 网络工具 / 文件处理 支持网页转 PDF 高效生成 PDF / 提供永久链接。 1. 产品功能 超高性能转换效率&#xff1b;支持将传递的 HTML 转换为 PDF&#xff0c;支持转换 HTML 中的 CSS 格式&#xff1b;支持传递网站 URL&#xff0c;直接转换页面成对应的 …

Java实现生成验证码实战

文章目录 需求描述思想思路实现代码实现效果 在实际项目中&#xff0c;管理端的登录&#xff0c;会涉及验证码的校验&#xff0c;简单的数字与字母组合形式&#xff0c;在Java中要如何生成与实现&#xff0c;记录下来&#xff0c;方便备查。 需求描述 生成8位的由数字、大写字…

【零基础学习CAPL】——CRC值监控测试

🙋‍♂️【零基础学习CAPL】系列💁‍♂️点击跳转 ——————————————————————————————————–—— 从0开始学习CANoe使用 从0开始学习车载车身 相信时间的力量 星光不负赶路者,时光不负有心人。 目录 1.概述2.需求介绍3.算法4.逻辑判断5.测…

ARCGIS PRO DSK MapTool

MapTool用于自定义地图操作工具&#xff0c;使用户能够在ArcGIS Pro中执行特定的地图交互操作。添加 打开MapTool1.vb文件&#xff0c;可以看到系统已经放出MapTool1类&#xff1a; Public Sub New()将 IsSketchTool 设置为 true 以使此属性生效IsSketchTool TrueSketchTyp…

为了准确计算延迟退休时间,我做了一个退休年龄计算器

延迟退休计算方法 原本退休分为三种情况&#xff0c;男性&#xff0c;女工人&#xff0c;女干部 男性&#xff1a;退休年龄为60岁。女干部&#xff1a;退休年龄为55岁。女工人&#xff1a;退休年龄为50岁。 现在延迟以后&#xff08;根据2024年9月13日公布的规则&#xff09…

一次开发,多端部署--实例二

一、视觉风格 1、分层参数 使用了分层参数后&#xff0c;当系统切换深色模式时&#xff0c;字体和背景也可以自适应。 Row() {Column() {Text(分层参数)// 分层参数在sysResource包&#xff0c;属于系统参数&#xff0c;全局可用.fontColor($r(sys_color.ohos_id_color_text_pr…

JavaScript模块化——ES6模块化规范

作者&#xff1a;CSDN-PleaSure乐事 欢迎大家阅读我的博客 希望大家喜欢 使用环境&#xff1a;vscode Chrome浏览器 1.ES6 1.1ES6介绍 ES6的全称是ECMAScript 6&#xff0c;也称为ES2015&#xff0c;是JavaScript的一个重要版本&#xff0c;它引入了许多新特性和改进&#xf…

云计算实训43——部署k8s基础环境、配置内核模块、基本组件安装

一、前期系统环境准备 1、关闭防火墙与selinux [rootk8s-master ~]# systemctl stop firewalld[rootk8s-master ~]# systemctl disable firewalldRemoved symlink /etc/systemd/system/multi-user.target.wants/firewalld.service. Removed symlink /etc/systemd/system/dbus…

云渲染与AI渲染分别是什么?两者的优势对比

云渲染和AI渲染是两种先进的渲染技术&#xff0c;它们各自具有独特的优势和应用场景。下面针对两种情况来简单说明下。 1、云渲染&#xff1a; - 定义&#xff1a;云渲染是一种利用远程服务器(云端)来处理和生成渲染效果的技术。它允许用户将计算密集型的任务转移到云端&#…

uniapp网络延迟优化之骨架屏

文章目录 前言uniapp网络延迟优化之骨架屏 一、骨架屏是什么&#xff1f;二、使用步骤1.在微信开发者工具生成骨架屏文件2.转成vue组件3.组件中使用4.效果展示4.开发时遇到的问题&#xff1f; 总结 前言 uniapp网络延迟优化之骨架屏 一、骨架屏是什么&#xff1f; 骨架屏的主…

C++ | Leetcode C++题解之第403题青蛙过河

题目&#xff1a; 题解&#xff1a; class Solution { public:bool canCross(vector<int>& stones) {int n stones.size();vector<vector<int>> dp(n, vector<int>(n));dp[0][0] true;for (int i 1; i < n; i) {if (stones[i] - stones[i -…

卷积神经网络经典模型架构简介

【图书推荐】《PyTorch深度学习与企业级项目实战》-CSDN博客 《PyTorch深度学习与企业级项目实战&#xff08;人工智能技术丛书&#xff09;》(宋立桓&#xff0c;宋立林)【摘要 书评 试读】- 京东图书 (jd.com) ImageNet是一个包含超过1 500万幅手工标记的高分辨率图像的数据…

Redis——常用数据类型List

目录 List列表常用命令lpushlpushxrpushrpushlrangelpoprpoplindexlinsertllenlremltrim key start stoplset 阻塞版本命令blpopbrpop list的编码方式list的应用 List列表 Redis中的list相当于数组&#xff0c;或者 顺序表&#xff0c;一些常用的操作可以通过下面这张图来理解…

ubuntu 遇到的一些问题及解决办法

一、“E: 无法定位软件包” 产生该问题的安全提示原因有很多。如网络链接问题、apt 源过期了。 解决方法&#xff1a; 1. 备份 /etc/apt/sources.list 文件 cp /etc/apt/sources.list /etc/apt/sources.list.old 2. 打开文件 gedit /etc/apt/sources.list 3. 添加清华源…

PHP一键约课高效健身智能健身管理系统小程序源码

一键约课&#xff0c;高效健身 —— 智能健身管理系统让健康触手可及 &#x1f3cb;️‍♀️ 告别繁琐&#xff0c;一键开启健身之旅 你还在为每次去健身房前的繁琐预约流程而烦恼吗&#xff1f;现在有了“一键约课高效健身智能健身管理系统”&#xff0c;所有问题都迎刃而解…

SaaS化多租户实现的两种方法

SaaS化多租户实现的两种方法 SaaS系统的定义 SaaS&#xff0c;全称为Software-as-a-Service&#xff08;软件即服务&#xff09;&#xff0c;是一种基于云计算的软件交付模式。而SaaS系统&#xff0c;即是通过这种模式提供给用户的软件系统。即多租户系统&#xff0c;每个租户…