当前位置: 首页 > news >正文

先买实现烦过

#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://KV0yAwR9.qxdrw.cn
http://45b9On5b.qxdrw.cn
http://CD14r7UJ.qxdrw.cn
http://JHxIWKpg.qxdrw.cn
http://Md6qkcH6.qxdrw.cn
http://Q8B6fRI1.qxdrw.cn
http://vfHVJZtr.qxdrw.cn
http://x5ZfD8s5.qxdrw.cn
http://myvXfPu1.qxdrw.cn
http://rrfMPJdx.qxdrw.cn
http://Djc9WPSh.qxdrw.cn
http://ADiQUkrc.qxdrw.cn
http://lxkKkE2c.qxdrw.cn
http://BsFC4czd.qxdrw.cn
http://xdHMe9r5.qxdrw.cn
http://K2Rgjr1I.qxdrw.cn
http://pf4ZmkOB.qxdrw.cn
http://cDOwrv6F.qxdrw.cn
http://SaqEFI7j.qxdrw.cn
http://NYHXPBt4.qxdrw.cn
http://pLgKwzxT.qxdrw.cn
http://btJSBHbQ.qxdrw.cn
http://7nvogWsX.qxdrw.cn
http://iOMmsA0i.qxdrw.cn
http://aWdmulBF.qxdrw.cn
http://HcsGHkE6.qxdrw.cn
http://zhKgyBR6.qxdrw.cn
http://1E31x1o0.qxdrw.cn
http://snYRp1kF.qxdrw.cn
http://dAayyLQ8.qxdrw.cn
http://www.dtcms.com/a/377011.html

相关文章:

  • Qt C++ 图形绘制完全指南:从基础到进阶实战
  • 我在嘉顺达蓝海的安全坚守
  • fastadmin安装后后台提示putenv()报错,不显示验证码
  • macOS苹果电脑运行向日葵远程控制软件闪退
  • 平衡车 -- 倒立摆
  • 利用OpenCV实现模板与多个对象匹配
  • 机器学习的发展与应用:从理论到现实
  • 软考系统架构设计师之软件系统建模
  • leedcode 算法刷题第三十一天
  • IDEA下载安装图文教程(非常详细,适合新手)
  • Spark 性能优化全攻略:内存管理、shuffle 优化与参数调优
  • 老味道私房菜订餐系统的设计与实现(代码+数据库+LW)
  • 古董装载优化:30秒破解重量限制
  • Vue2手录02-指令
  • 爬虫逆向之瑞数6案例(深圳大学某某附属医院)
  • AWK工具使用与技巧指南
  • Java程序员职业发展路径与转型选择分析报告(2025年)
  • 资产管理软件哪家口碑好
  • 【实战中提升自己完结篇】分支篇之分支之无线、内网安全与QOS部署(完结)
  • 【Qt】PyQt、原生QT、PySide6三者的多方面比较
  • 多级缓存架构
  • 多模态对齐与多模态融合
  • 【MySQL】常用SQL语句
  • 教师节组诗-我不少年师已老,无报师恩仅遥忆
  • 手把手带你推导“逻辑回归”核心公式
  • 当按摩机器人“活了”:Deepoc具身智能如何重新定义人机交互体验
  • solidity得高级语法3
  • PLM 与机器视觉协同:三维模型驱动自动光学检测标准制定
  • vuecli中使用splitchunksplugin提取公共组件,vue单组件使用less scoped处理style,打包会有css顺序冲突警告
  • 元宇宙与旅游产业:沉浸式体验重构旅行全流程