网络编程:TCP机械臂,UDP文件传输

1.TCP机械臂测试

程序代码:

  1 #include<myhead.h>
  2 #define SER_IP "192.168.126.112"  //服务器IP
  3 #define SER_PORT 8888          //服务器端口号
  4 
  5 #define CLI_IP "192.168.126.121" //客户端IP
  6 #define CLI_PORT 9999       //客户端端口号
  7 int main(int argc, const char *argv[])
  8 {
  9     //1.创建用于通信的套接字文件描述符
 10     int cfd=socket(AF_INET,SOCK_STREAM,0);  //通信协议族,IPv4;指定通信类型,TCP;参数2已指定通信类型
 11     if(cfd==-1)
 12     {
 13         perror("socket error");
 14         return -1;
 15     }
 16     printf("cfd=%d\n",cfd);
 17     //2.绑定(非必须)
 18     //2.1填充地址信息结构体
 19     struct sockaddr_in cin;
 20     cin.sin_family=AF_INET;//地址族,IPv4
 21     cin.sin_port=htons(CLI_PORT);//端口号,主机号转换为网络号,2字节
 22     cin.sin_addr.s_addr=inet_addr(CLI_IP);//ip地址,点分十进制转为4字节无符号网络字节序
 23     //2.2绑定,将ip地址和端口号绑定到套接字文件描述符
 24     if(bind(cfd,(struct sockaddr*)&cin,sizeof(cin))==-1)
 25         //文件描述符;
 26         //通信地址信息结构体,不同通信方式,结构体不同,将具体地址信息强转为该类型,地址
 27         //结构体大小,值
 28     {
 29         perror("bind error");
 30         return -1;
 31     }
 32     printf("bind success\n");
 33     //3.连接服务器
 34     //3.1填充要连接的服务器地址信息结构体
 35     struct sockaddr_in sin;
 36     sin.sin_family=AF_INET;//地址族IPv4
 37     sin.sin_port=htons(SER_PORT);//端口号,主机号转换为网络号
 38     sin.sin_addr.s_addr=inet_addr(SER_IP);//ip地址,点分十进制转为4字节无符号网络字节序
 39     //3.2连接服务器
 40     if(connect(cfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
 41         //套接字文件描述符
 42         //要发送数据起始地址
 43         //要发送数据大小
 44         //是否阻塞,0阻,MSG_DONTWAIT非阻塞
 45     {
 46         perror("connect error");
 47         return -1;
 48     }
 49     printf("connect success\n");
 50     //4.数据收发
 51     char rbuf[5]={0xff,0x02,0x00,0x00,0xff};
 52     //起始结束协议,固定的;控制机械臂协议,固定的
 53     //x指定操作机械臂,0x00红,0x01蓝;y:指定角度
 54     unsigned char bbuf[5]={0xff,0x02,0x01,0x00,0xff};
 55     //发送给服务器当做初始值
 56     send(cfd,rbuf,sizeof(rbuf),0);//文件描述符;要发送数据起始地址;数据大小;是否阻塞
 57     sleep(1);
 58     send(cfd,bbuf,sizeof(bbuf),0);
 59     char key=0;//接收键盘输入的字符
 60     while(1)
 61     {
 62         system("clear");//执行终端指令
 63         scanf("%c",&key);//键盘输入一个字符
 64         getchar();//吸收垃圾字符,回车
 65         switch(key)
 66         {
 67             //红色机械臂
 68         case 'W':
 69         case 'w':
 70             {
 71                 rbuf[3]+=5;//每次操作角度增加5度
 72                 if(rbuf[3]>=90)
 73                     rbuf[3]=90;
 74                 send(cfd,rbuf,sizeof(rbuf),0);
 75             }
 76             break;
 77         case 'S':
 78         case 's':
 79             {
 80                 rbuf[3]-=5;//每次操作角度减小5度
 81                 if(rbuf[3]<=-90)
 82                     rbuf[3]=-90;
 83                 send(cfd,rbuf,sizeof(rbuf),0);
 84             }
 85             break;
 86             //蓝色机械臂
 87         case 'D':
 88         case 'd':
 89             {
 90                 bbuf[3]+=5;//每次操作角度增加
 91                 if(bbuf[3]>=180)
 92                     bbuf[3]=180;
 93                 send(cfd,bbuf,sizeof(bbuf),0);
 94             }
 95             break;
 96         case 'A':
 97         case 'a':
 98             {
 99                 bbuf[3]-=5;//每次操作角度减小
100                 if(bbuf[3]<=0)
101                     bbuf[3]=0;
102                 send(cfd,bbuf,sizeof(bbuf),0);
103             }
104             break;
105         case 'Q':
106         case 'q':
107             goto END;
108         default:printf("enter error\n");
109         }
110     }
111 END:
112     //5.关闭套接字
113     close(cfd);
114     return 0;
115 }                                                                                                                                                                                                                                                      
~                                                                                                                                                                                                                                                          
~                

运行结果:

2. 基于UDP的TFTP文件传输

程序代码:

#include <myhead.h>
#define SER_IP "192.168.125.254"        //服务端IP
#define SER_PORE 69                     //端口号
 
//定义下载函数
int download(int cfd,struct sockaddr_in sin)
{
		
	//组件协议包:下载请求
	char pack[516]="";
	short *p1=(short *)pack;
	*p1=htons(1);          //设置操作码
 
	char file[20]="";
	printf("请输入文件名:");
	fgets(file,sizeof(file),stdin);
	file[strlen(file)-1]=0;   
 
	char *p2=(pack+2);
	strcpy(p2,file);   //文件名
 
	char *p4=p2+strlen(p2)+1;
	strcpy(p4,"octet");        //模式位
 
	int packlen=4+strlen(p2)+strlen(p4);  //请求包大小
	printf("%d\n",packlen);
 
	//向服务器发送请求包
	sendto(cfd,pack,packlen,0,(struct sockaddr*)&sin,sizeof(sin));
 
	//以只写的形式打开文件
	int wfd=-1;
	if((wfd=open(file,O_WRONLY|O_CREAT|O_TRUNC,0664))==-1){
		perror("open error");
		return -1;
	}
	int res=0;
 
	struct sockaddr_in cin;
	socklen_t socklen=sizeof(cin);
 
	while(1)
	{
		//收取服务器发来的数据包
		//判断数据包的操作码是否为3 以及 数据包数据内容是否为512字节
		//若为512字节,则读取后,回复一个应答包,继续接收下一个
		//如果小于512,则读取数据后,回复一个应答包,结束接收数据
 
		//清空协议包
		bzero(pack,sizeof(pack));
		
		res=recvfrom(cfd,pack,sizeof(pack),0,(struct sockaddr*)&cin,&socklen); //接收数据包
		
		if(*p1==ntohs(3))         //数据包
		{
			write(wfd,pack+4,res-4);
			//回复应答包
			*p1=htons(4);
			sendto(cfd,pack,4,0,(struct sockaddr*)&cin,sizeof(cin));
 
			if(res-4<512){
				printf("下载完成\n");
				break;
			}
		}else if(*p1==ntohs(5))      //错误包
		{
			printf("error\n");
			return -1;
		}
	}
	close(wfd);
	return 0;
}
 
//定义上传函数
int upload(int cfd,struct sockaddr_in sin)
{
	//组件协议包:上传请求
	char pack[516]="";
	unsigned short *p1=(unsigned short *)pack;
	*p1=htons(2);          //设置操作码
 
	char file[20]="";
	printf("请输入文件名:");
	fgets(file,sizeof(file),stdin);
	file[strlen(file)-1]=0;   
 
	char *p2=(pack+2);
	strcpy(p2,file);   //文件名
 
	char *p4=p2+strlen(p2)+1;
	strcpy(p4,"octet");        //模式位
 
	int packlen=4+strlen(p2)+strlen(p4);  //请求包大小
	printf("%d\n",packlen);
 
	//向服务器发送请求包
	sendto(cfd,pack,packlen,0,(struct sockaddr*)&sin,sizeof(sin));
 
	//以只读的形式打开文件
	int rfd=-1;
	if((rfd=open(file,O_RDONLY,0664))==-1){
		perror("open error");
		return -1;
	}
	int res=0;
 
	struct sockaddr_in cin;
	socklen_t socklen=sizeof(cin);
 
	unsigned short num=0;         //定义块编号;
	while(1)
	{
		
		//清空
		bzero(pack,sizeof(pack));
 
		res=recvfrom(cfd,pack,sizeof(pack),0,(struct sockaddr*)&cin,&socklen);
		
		if(*p1==ntohs(4))         //接收应答包
		{
			//发送数据包
			*p1=htons(3);
 
			//填充块编号
			num++;
			*(p1+1)=htons(num);
			//读取数据
			res=read(rfd,p1+2,512);
			sendto(cfd,pack,sizeof(pack),0,(struct sockaddr*)&cin,sizeof(cin));
			if(res==0){
				printf("上传完成\n");
				break;
			}else if(res<0){
				printf("error\n");
				return -1;
			}
		}else if(*p1==ntohs(5))      //错误包
		{
			printf("error\n");
			return -1;
		}
	}
	close(rfd);
	return 0;
}

int main(int argc, const char *argv[])
{
	/*
	//判断是否外部传参
	if(argc!=2){
	printf("unknow filename\n");
	return -1;
	}
	*/
	//1.创建用于通信的套接字文件描述符
	int cfd=-1;
	if((cfd=socket(AF_INET,SOCK_DGRAM,0))==-1){
		perror("socket error");
		return -1;
	}
	//2.数据收发
	//填充服务器的地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(SER_PORE);
	sin.sin_addr.s_addr=inet_addr(SER_IP);
 
	//创建菜单
	int menu=0;
	while(1)
	{
		system("clear");              //执行终端指令
 
		printf("\t******1.下载******\n");
		printf("\t******2.上传********\n");
		printf("\t******3.退出******\n");
		printf("请输入选项:");
		scanf("%d",&menu);
		getchar();//吸收垃圾字符,回车
		switch(menu)
		{
		case 1:
			{
				//下载
				download(cfd,sin);				
			}
			break;
		case 2:
			{
				//上传
				upload(cfd,sin);
			}
			break;
		case 3:
			{
				//退出
				goto END;
 
			}
			break;
		default:printf("enter error\n");break;
		}
	}
 
END:
	//3.关闭套接字
	close(cfd);
 
	return 0;
}

运行结果:

 流程图:

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

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

相关文章

Microsoft PyRIT能自动化完成AI红队的任务

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

web3时事粥报

比特币正成为更具有吸引力的通胀对冲工具 在通胀的宏观经济浪潮中&#xff0c;比特币正逐渐崭露头角&#xff0c;成为那些渴望多元化投资组合的投资者眼中的璀璨明星。Kooner 预测&#xff0c;2024年&#xff0c;各种宏观经济挑战可能进一步提升比特币、黄金和白银等资产的避险…

群体风暴之锤(War3地图编辑器)

文章目录 0、大致原理1、创建隐形单位2、新事件开端3、环境→新条件4、动作4.1、单位组4.1.1、圆范围内单位4.1.2、指定条件 4.2、对单位组内的所有单位释放风暴之锤 0、大致原理 真MK向目标点释放风暴之锤时选定&#xff08;以技能释放点为圆心&#xff0c;设定半径&#xff0…

【RT-DETR有效改进】结合SOTA思想利用双主干网络改进RT-DETR(全网独家创新,重磅更新)

一、本文介绍 本文给大家带来的改进机制是结合目前SOTAYOLOv9的思想利用双主干网络来改进RT-DETR&#xff08;本专栏目前发布以来改进最大的内容&#xff0c;同时本文内容为我个人一手整理全网独家首发 | 就连V9官方不支持的模型宽度和深度修改我都均已提供&#xff0c;本文内…

JUC并发编程 深入学习Java并发编程【上】

JUC并发编程&#xff0c;深入学习Java并发编程&#xff0c;与视频每一P对应&#xff0c;全系列6w字。 P1-5 为什么学特色预备知识 进程线程概念 进程&#xff1a; 一个程序被运行&#xff0c;从磁盘加载这个程序的代码到内存&#xff0c;就开起了一个进程。 进程可以视为程…

搜索旋转排序数组[中等]

优质博文IT-BLOG-CN 一、题目 整数数组nums按升序排列&#xff0c;数组中的值 互不相同 。 在传递给函数之前&#xff0c;nums在预先未知的某个下标k&#xff08;0 < k < nums.length&#xff09;上进行了 旋转&#xff0c;使数组变为[nums[k], nums[k1], ..., nums[n-…

C语言冒泡排序(高级版)

目录: 冒泡排序的原理 主函数 "冒泡排序函数" 比较函数 交换函数 最终输出 完整代码 冒泡排序的原理: 冒泡排序的原理是&#xff1a;从左到右&#xff0c;相邻元素进行比较。每次比较一轮&#xff0c;就会找到序列中最大的一个或最小的一个。这个数就会从序列的最右…

Qt 简约美观的加载动画 第九季

这次和大家分享6个非常清爽的加载动画. &#x1f60a; 效果如下 &#x1f60a; 一共三个文件 , 可以直接编译运行的呢 //main.cpp #include "LoadingAnimWidget.h" #include <QApplication> #include <QGridLayout> int main(int argc, char *argv[]) …

【机器人最短路径规划问题(栅格地图)】基于模拟退火算法求解

代码获取方式&#xff1a;QQ&#xff1a;491052175 或者 私聊博主获取 基于模拟退火算法求解机器人最短路径规划问题&#xff08;栅格地图&#xff09;的仿真结果 仿真结果&#xff1a; 初始解的路径规划图 收敛曲线&#xff1a; 模拟退火算法求解的路径规划图 结论&#xff…

DVWA靶场 Command Injection,高中低

Low 输入ip地址正常显示&#xff0c;尝试加入其他命令 127.0.0.1 & whoami 后面的whoami也执行了 Medium whoami也可以执行 好像&可应用&#xff0c;&&应该是被过滤 High &用不了&#xff0c;应该是过滤了吧 经过尝试&、|都无法用 查看源码后发现有…

GO逃逸分析

内存管理 内存管理主要包括两个动作&#xff1a;分配与释放。逃逸分析就是服务于内存分配的&#xff0c;而内存的释放由GC负责。 栈 在Go语言中&#xff0c;栈的内存是由编译器自动进行分配和释放的&#xff0c;栈区往往存储着函数参数、局部变量和调用函数帧&#xff0c;它…

Java 计算某年份二月的天数

一、实验任务 要求编写一个程序&#xff0c;从键盘输入年份&#xff0c;根据输入的年份计算这一年的2月有多少天。 二、实验内容 三、实验结果 四、实现逻辑和步骤 &#xff08;1&#xff09;使用scanner类实现程序使用键盘录入一个年份。 &#xff08;2&#xff09;使用if语…

百度诉闪速推公司涉“万词霸屏”不正当竞争纠纷案审理结果

交叉口讯 5月13日&#xff0c;江苏省高级人民法院知识产权庭公布百度诉闪推公司涉及“万磁霸屏”不正当竞争纠纷一案审理结果&#xff1a;判决闪推公司应立即停止涉案的不正当竞争行为。 &#xff0c;公司在其公司官网发布声明&#xff0c;消除影响&#xff0c;并赔偿百度经济损…

Pandas DataFrame 基本操作实例100个

Pandas 是一个基于NumPy的数据分析模块&#xff0c;最初由AQR Capital Management于2008年4月开发&#xff0c;并于2009年底开源。Pandas的名称来源于“Panel Data”&#xff08;面板数据&#xff09;和“Python数据分析”&#xff08;data analysis&#xff09;。这个库现在由…

Google Dremel和parquet的复杂嵌套数据结构表征方法解析

转载请注明出处。作者&#xff1a;archimekai 核心参考文献&#xff1a; Dremel: Interactive Analysis of Web-Scale Datasets 文章目录 引言复杂嵌套数据结构的无损表征问题Dremel论文中提出的表征方法parquet备注 引言 Dremel是Google的交互式分析系统。Google大量采用prot…

IDEA POM文件配置profile实现不同环境切换

目录 一、背景 二、实现 2.1创建不同的配置文件 2.2配置POM文件 三、效果 3.1本地使用 2.2线上或者测试环境使用 一、背景 在企业级开发中&#xff0c;为了不影响生产环境的项目运行&#xff0c;一般情况下都会划分生产环境、测试环境、开发环境。不同环境可以配置不同的…

ubuntu安裝Avahi发现服务工具

一、简介 解决设置固定ip后无法连接外网的问题&#xff0c;目前采用动态获取ip&#xff0c;可以不用设置设备的固定IP&#xff0c;直接可以通过域名来访问设备&#xff0c;类似树莓派的连接调试 二、安装 本文使用的是ubuntu23.10.1上安装 1.安装工具 sudo apt install av…

ABAP - SALV教程17 弹窗ALV

SALV可以通过弹窗形式打开在生成SALV实例对象后调用set_screen_popup方法设置成弹出模式 "设置为弹窗模式 go_alv->set_screen_popup( start_column 10end_column 110start_line 5end_line 15). 显示效果 完整代码 SELECT *FROM ekkoINTO TABLE DATA(gt_dat…

使用plasmo框架开发浏览器插件,注入contents脚本和给页面添加UI组件

plasmo&#xff1a;GitHub - PlasmoHQ/plasmo: &#x1f9e9; The Browser Extension Framework plasmo是一个开发浏览器插件的框架&#xff0c;支持使用react和vue等技术&#xff0c;而且不用手动管理manifest.json文件&#xff0c;框架会根据你在框架中的使用&#xff0c;自…

二极管原理及典型应用电路、三极管基本结构及类型状态

目录 二极管原理及典型应用电路 二极管的工作原理 二极管保护电路 二极管整流电路 二极管稳压电路 三极管基本结构及类型状态 三极管基本结构和类型 三极管的 3 种工作状态 二极管原理及典型应用电路 如下图&#xff0c;二极管长成这样。它们通常有一个黑色圆柱体&am…