先買實現煩過

#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
英文地址,請注明出處:http://en.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…