先买实现烦过

#include <myhead.h>
#define ERR_LOG(msg)do{perror(msg);printf("%d %s %s\n",__LINE__,__func__,__FILE__);}while(0)
//定义TFTP默认端口号(69)和数据包大小(516字节)
#define PORT    69
#define N     516
int main(int argc, const char *argv[])
{
//检查命令行参数,确保提供了服务器的IP地址
if(argc<2)
{
printf("请输入IP\n");
return -1;
}
//创建UDP套接字文件描述符
int sfd=socket(AF_INET,SOCK_DGRAM,0);
if(sfd<0)
{
perror("socket:");
return -1;
}
//填充服务器地址结构
struct sockaddr_in sin;
sin.sin_family=AF_INET;
sin.sin_addr.s_addr=inet_addr(argv[1]);//IP地址转换为网络字节序
sin.sin_port=htons(PORT);               //端口号转换为网络字节序
//主循环
char choose;//选择的容器变量
while(1)
{
system("clear");//清屏
//显示功能菜单
printf("************菜单************\n");
printf("**********1:下载**********\n");
printf("**********2:上传**********\n");
printf("**********3:退出**********\n");
printf("请输入您要选择的功能:");
//用户选择
scanf("%c",&choose);
//清空输入缓冲区
while(getchar()!='\n');
//根据用户选择执行相应的功能
switch(choose)
{
case '1':
//下载自定义函数
xia(sfd,sin);
break;
case '2':
//上传自定义函数
chuan(sfd,sin);
break;
case '3':
goto END;
break;
default:
printf("输入错误\n");
}
printf("请输入任意键清屏");
while(getchar()!='\n');
}
END:
close(sfd);
return 0;
}
//下载函数
int xia(int sfd,struct sockaddr_in sin)
{
//获取用户要下载的文件名
char name[20]="";
printf("请输入要下载的文件名:");
fgets(name,20,stdin);//从终端输入20字节写入到name里
name[strlen(name)-1]=0;//去掉换行符
//发送下载请求
char buf[N]="";//定义一个请求包大小的储存容器
//构建TFTP读请求包(操作码1)
int size=sprintf(buf,"%c%c%s%c%s%c",0,1,name,0,"octet",0);
//%c%c:前两个字节为操作码,这里传入0, 1表示 "读请求"(下载操作)
//%s:紧跟操作码的是文件名(变量filename)
//%c:文件名后用空字符0作为分隔符
//%s:接下来是传输模式字符串 "octet"(二进制模式)
//最后的%c:模式字符串后也用空字符0结尾
//发送请求到服务器
if(sendto(sfd,buf,size,0,(struct sockaddr *)&sin,sizeof(sin))<0)
{
ERR_LOG("sendto");
return -1;
}
//循环接收数据并处理
int flag=0;//标记文件是否已创建
int fd;//本地文件描述符(用于保存下载的文件)
ssize_t recv_len;       //接收的字节字数
unsigned short num =1;  //期望接收的块编号(从1开始)
socklen_t addrlen =sizeof(sin);
while(1)
{
bzero(buf,N);//清空缓冲区
//接收服务器的相应
recv_len=recvfrom(sfd,buf,N,0,(struct sockaddr *)&sin,&addrlen);
if(recv_len<0)
{
ERR_LOG("recv_len:");
return -1;
}
//处理服务器发送的数据包(操作码3)
if(3==buf[1])//操作码3表示[数据块]
{
if(0==flag)//首次接收数据,就创建本地文件
{
fd=open(name,O_WRONLY|O_CREAT|O_TRUNC,0664);
if(fd<0)
{
ERR_LOG("open:");
return -1;
}
flag=1;//标记文件已创建
}
//验证数据块编号(防丢包/重复)
//检查接收的块编号是否与期望的一致
if(htons(num)==*(unsigned short *)(buf+2))
{
//将数据块中的内容跳过头部写入本地文件
if(write(fd,buf+4,recv_len-4)<0)
{
ERR_LOG("write:");
break;
}
buf[1]=4;//将操作吗改成4(确认包)
if(sendto(sfd,buf,4,0,(struct sockaddr *)&sin,sizeof(sin))<0)
{
ERR_LOG("sendto");
}
//如果数据小于516字节呢就说明是最后一个
if(recv_len<516)
{
printf("下载完毕");
break;
}
num++;//准备接收下一个块
}
}else if(5==buf[1])//收到错误包
{
printf("ERROR:%s",buf+4);//打印错误信息
break;
}
}
return 0
}
int chuan(int sfd, struct sockaddr_in sin)
{
char filename[20] = "";
printf("请输入要上传的文件名>>>");
fgets(filename, 20, stdin);
filename[strlen(filename)-1] = 0;
//判断该文件是否存在
int fd = open(filename, O_RDONLY);
if(fd < 0)
{
if(errno == ENOENT)
{
printf(">>>文件不存在,请重新输入<<<\n");
return -2;
}
else
{
ERR_LOG("open");
return -1;
}
}
//发送上传请求
//上传协议
char buf[N] = "";
int size = sprintf(buf, "%c%c%s%c%s%c", 0, 2, name, 0, "octet", 0);
if(sendto(sfd, buf, size, 0, (struct sockaddr*)&sin, sizeof(sin))<0)
{
ERR_LOG("sendto");
return -1;
}
//循环接收发送数据包
int recv_len;
unsigned short num = 0;
socklen_t addrlen = sizeof(sin);
while(1)
{
bzero(buf, N);
recv_len = recvfrom(sfd, buf, sizeof(buf), 0, (struct sockaddr*)&sin, &addrlen);
if(recv_len < 0)
{
ERR_LOG("recvfrom");
return -1;
}

        //操作码的范围是1-5,因为是网络字节序
//所以有效操作吗存储在高位,即buf[1]的位置.
//printf("buf[1] = %d\n", buf[1]);     //4 服务器返回应答包
if(4 == buf[1])
{
//判断当前数据包的编号是否等于应答包的编号
//防止数据包在传送过程丢包或重复收包
if(num == ntohs(*(unsigned short*)(buf+2)))
{
//修改操作码为数据包
buf[1] = 3;
//填充块编号
num++;
*(unsigned short*)(buf+2) = htons(num);
//读取数据
//发送数据
int res = read(fd, buf+4, N-4);
if(res < 0)
{
ERR_LOG("read");
return -1;
}
else if(0 == res)
{
printf("-----文件上传完毕-----\n");
break;
}
//发送数据包
//发送的数据包大小为,读取到的字节数(res)+操作码(2byte)+快编号(2byte)
if(sendto(sfd, buf, res+4, 0, (struct sockaddr*)&sin, sizeof(sin)) <0)
{
ERR_LOG("sendto");
return -1;
}
}
else
{
printf("-----文件上传失败,请检查网络环境-----\n");
break;
}
}
else if(5 == buf[1])
{
printf("-----ERROR:%s-----\n", buf+4);
break;
}
}
return 0;
}

#include <myhead.h>
#define SER_IP "192.168.108.124"//服务器IP地址
#define SER_PORT 8888//服务器端口号
#define CLT_IP "192.168.24.128"//客户端IP地址
#define CLT_PORT 7777//客户端端口号
int main(int argc, const char *argv[])
{
//创建用于连接的套接字文件描述符
int cfd=socket(AF_INET,SOCK_STREAM,0);
if(cfd==-1)
{
perror("scoket error:");
return -1;
}
//打开键盘驱动文件
int fd=open("/dev/input/event1",O_RDONLY);
if(fd==-1)
{
perror("open error:");
return -1;
}
//定义容器接收数据
struct input_event ie;
//绑定IP地址和端口号
//填充地址信息结构体
struct sockaddr_in cin;
cin.sin_family=AF_INET;
cin.sin_addr.s_addr=inet_addr(CLT_IP);
cin.sin_port=htons(CLT_PORT);
//绑定
if(bind(cfd,(struct sockaddr *)&cin,sizeof(cin))==-1)
{
perror("bind error:");
return -1;
}
//连接到服务器
//填充服务器地址信息结构体
struct sockaddr_in sin;
sin.sin_family=AF_INET;
sin.sin_addr.s_addr=inet_addr(SER_IP);
sin.sin_port=htons(SER_PORT);
//连接服务器
if(connect(cfd,(struct sockaddr *)&sin,sizeof(sin))==-1);
{
perror("connect error:");
return -1;
}
printf("连接成功");

    char rbuf[5]={0xff,0x02,0x00,0x00,0xff};
unsigned char bbuf[5]={0xff,0x02,0x01,0x00,0xff};
//发送给服务器,初始化机械臂
send(cfd,rbuf,sizeof(rbuf),0);
sleep(1);
send(cfd,bbuf,sizeof(bbuf),0);
while(1)
{
//从文件中读取数据
read(fd,&ie,sizeof(ie));
//对输入的数据判断
switch(ie.code*ie.value)
{
case 17://W
{
bbuf[3]+=2;
if(bbuf[3]>180)
{
bbuf[3]=180;
}
//将更新过后的数据发送给服务器
send(cfd,bbuf,sizeof(bbuf),0);
}
break;
case 31://S
{
bbuf[3]-=2;
if(bbuf[3]<0)
{
bbuf[3]=0;
}
//将更新过后的数据发送给服务器
send(cfd,bbuf,sizeof(bbuf),0);
}
break;
case 30://A
{
rbuf[3]+=2;               
if(rbuf[3]>90)
{
rbuf[3]=90;
}   
//将更新过后的数据发送给服务器
send(cfd,rbuf,sizeof(rbuf),0);
}     
break;
case 32://D
{
rbuf[3]-=2;               
if(rbuf[3]<0)
{
rbuf[3]=0;
}   
//将更新过后的数据发送给服务器
send(cfd,rbuf,sizeof(rbuf),0);
}     
break;
}
}
close(cfd);
return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ACD智能分配:轮流分配和排序上限分配的设置

在客户服务中&#xff0c;合理的对话分配是提高服务质量的关键。一洽客服系统针对不同业务场景,提供灵活的客服分配策略,帮助企业实现智能化的客户服务管理&#xff0c;今天我们了解一下对话的轮流分配、排序上限分配、排序优先分配的设置一、轮流分配按照客服登录系统的先后顺…

【postMan / apifox 文件上传】

apifox 需要提供相关插件 失败的请求 { “timestamp”: “2025-09-10T14:44:24.91900:00”, “status”: 500, “error”: “Internal Server Error”, “path”: “/student/import” } 错误&#xff1a;Post “http://localhost:8080/student/import”: dial tcp [::1]:8080:…

视频加水印,推荐使用运营大管家-视频批量加水印软件

运营大管家-视频批量加水印软件介绍“运营大管家-视频批量加水印”是一款功能强大的桌面应用程序&#xff0c;旨在帮助用户高效地为多个视频批量添加自定义水印。无论是品牌宣传、版权保护&#xff0c;还是个性化展示&#xff0c;本软件都能提供灵活的文字水印和图片水印选项&a…

基于 Dockerfile 构建镜像

1.准备构建上下文[roothost1 ~]# mkdir dockerfile-test && cd dockerfile-test [roothost1 dockerfile-test]# touch nginx.repo [roothost1 dockerfile-test]# touch Dockerfile [roothost1 dockerfile-test]# vi nginx.repo [roothost1 dockerfile-test]# cat nginx…

[Dify实战]插件编写- 如何让插件直接输出文件对象(支持 TXT、Excel 等)

在大多数 Dify 插件开发中,我们习惯于让插件返回结构化文本、字典或 JSON 数据。但随着应用场景拓展,例如翻译文件、生成报表、处理数据分析结果等,我们需要让插件支持“直接返回文件对象”给用户,而不是让用户复制粘贴文本再手动保存。 本文将基于实战经验,详细介绍如何…

Thread类的基本用法(上)

一、线程创建方法&#xff08;5种&#xff09;1.继承Thread类class MyThread extends Thread {Overridepublic void run() {System.out.println("MyThread is running");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}…

ARM内存映射与启动地址重映射机制解析

目录 内存映射 1. 核心概念&#xff1a;内存映射 (Memory Map) 2. 启动过程与地址重映射 (Remapping) 关键&#xff1a;启动引脚 (Boot Pins) 这个过程可以类比&#xff1a; 3. 为什么设计成这样&#xff1f; 4. 一图流总结 图解说明&#xff1a; 核心要点&#xff1a;…

网络原理——传输层协议TCP基本认识

文章目录传输层协议TCP基本认识TCP协议的格式TCP的可靠性初步理解——确认应答机制暂时理解TCP的通信过程TCP的确认号和确认序号确认号和确认序号的意义捎带应答TCP中其他字段的理解16位窗口大小标志位标志位的本质标志位的意义以SYN ACK标志位简单理解TCP连接三次握手以FIN标…

Java HTTP响应的流式处理技术

第1章 引言 1.1 传统HTTP响应处理的局限性 在现代Web应用开发中,HTTP通信是系统间数据交换的核心方式。随着数据量的不断增长和实时性要求的提高,传统的HTTP响应处理方式逐渐暴露出诸多问题。 传统处理方式通常需要将整个HTTP响应体一次性加载到内存中,然后再进行处理。这…

D01-【计算机二级】Python(1)基本操作第41题

1、考生文件夹下存在一个文件 PY101.py&#xff0c;请写代码替换横线&#xff0c;不修改其他代码&#xff0c;实现以下功能: 键盘输入正整数 n&#xff0c;按要求把 n 输出到屏幕&#xff0c;格式要求&#xff1a;宽度为 20 个字符&#xff0c;减号字符 - 填充&#xff0c;右对…

工程师 - Onion Architecture in Software Development

Introduction 介绍 In the ever-evolving world of software development, finding the right architectural pattern is akin to selecting the foundation for a building. One such architectural paradigm that has gained recognition for its ability to promote mainta…

TightVNC功能介绍

TightVNC是一款跨平台的远程桌面工具&#xff0c;支持Windows、Linux等系统&#xff0c;通过高效压缩技术实现低带宽环境下的流畅控制。以下是详细的使用说明&#xff1a; 一、安装与配置 1. Windows系统 下载与安装 访问TightVNC官网下载安装包&#xff0c;运行后选择“Comp…

硬件 (七) ARM 软中断, IMX6ULL 点灯

一、ARM 软中断&#xff08;SVC&#xff09;&#xff1a;从用户态到内核态的桥梁软中断&#xff08;SVC&#xff0c;Supervisor Call&#xff09;是 ARM 处理器从 “非特权模式&#xff08;如 User&#xff09;” 进入 “特权模式&#xff08;如 Supervisor&#xff09;” 的核…

数据结构与算法-树和二叉树-二叉树的存储结构(Binary Tree)

树和二叉树的内容比较多&#xff0c;分成两次来发 4 树和二叉树&#xff08;Tree and Binary Tree&#xff09; 4.1 树和二叉树的定义 4.1.1 树的定义 树&#xff08;Tree&#xff09;是 n&#xff08;n>0&#xff09;个结点的有限集&#xff0c;它或为空树&#xff08;…

CentOS7 Hive2.3.8 安装图文教程

一、 安装MySQL 0.0&#xff09;查询mariadb,有就去0.1&#xff09;&#xff0c;没有就不管直接去1&#xff09; rpm -qa | grep mariadb0.1&#xff09;卸载mariadb rpm -e --nodeps 查询出来的内容二、安装MySQL 1.下载资源包 官网下载 MySQL官网下载地址:https://dev.m…

开发避坑指南(43):idea2025.1.3版本启动springboot服务输入jvm参数解决办法

问题 最近装了新版IDEA尝尝新特性&#xff0c;IntelliJ IDEA 2025.1.3&#xff0c;可是在运行springboot服务的时候&#xff0c;找了好久才找到输入jvm启动参数的地方。如上图&#xff0c;不像旧版的IDEA&#xff0c;在Run/Debug Configurations->Configuration->Environ…

《sklearn机器学习——数据预处理》标准化或均值去除和方差缩放

数据集的标准化是scikit-learn中实现许多机器学习估计器的普遍要求&#xff1b;如果个别特征看起来或多或少不像标准正态分布数据&#xff1a;均值和单位方差为零的高斯分布&#xff0c;则它们的性能可能不好。 在实践中&#xff0c;我们通常会忽略分布的形状&#xff0c;而只是…

leedcode 算法刷题第三十一天

1049. 最后一块石头的重量 II 有一堆石头&#xff0c;用整数数组 stones 表示。其中 stones[i] 表示第 i 块石头的重量。 每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将它们一起粉碎。假设石头的重量分别为 x 和 y&#xff0c;且 x < y。那么粉碎的可能结果…

图神经网络介绍

源自论文&#xff1a;Survey on Graph Neural Networks 图神经网络&#xff08;GNNs&#xff09;中的符号与定义详解 本文使用了图论和深度学习领域的标准符号体系&#xff0c;以确保对图结构数据的描述清晰一致。以下是核心符号和定义的详细说明&#xff1a; 一、基础图结构符…

测试报告:“问卷考试系统”项目

目录 一、报告概述 &#xff08;一&#xff09;项目背景 &#xff08;二&#xff09;项目核心模块与测试目的 1、项目核心模块 2、测试目的 &#xff08;三&#xff09;测试环境 1、硬件环境 2、软件环境 &#xff08;1&#xff09;操作系统 &#xff08;2&#xff0…