Day34 UDP套接字編程 可靠文件傳輸與實時雙向聊天系統

day34 UDP套接字編程 可靠文件傳輸與實時雙向聊天系統

UDP文件傳輸

實現客戶端向服務器傳輸文件(如圖片)的功能,確保傳輸后文件內容完全一致且可正常打開。傳輸過程采用簡單的確認機制防止數據包丟失,傳輸完成后雙方程序自動退出。核心知識點包括:UDP套接字創建、文件讀寫、數據包分塊傳輸、傳輸結束標志處理。

客戶端代碼 (cli.c)

#include <arpa/inet.h>   // 提供IP地址轉換函數
#include <fcntl.h>       // 提供文件控制操作
#include <netinet/in.h>  // 定義IPv4地址結構
#include <netinet/ip.h>  // 定義IP協議相關結構
#include <stdio.h>       // 標準輸入輸出函數
#include <stdlib.h>      // 標準庫函數
#include <string.h>      // 字符串操作函數
#include <sys/socket.h>  // 套接字API
#include <sys/types.h>   // 數據類型定義
#include <time.h>        // 時間相關函數
#include <unistd.h>      // POSIX系統調用typedef struct sockaddr *(SA);  // 定義sockaddr指針別名,簡化類型轉換int main(int argc, char **argv)
{// 創建UDP套接字(AF_INET: IPv4, SOCK_DGRAM: 數據報套接字)int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (-1 == sockfd){perror("socket");  // 套接字創建失敗時打印錯誤return 1;}// 配置服務器地址結構struct sockaddr_in ser;ser.sin_family = AF_INET;         // 地址族:IPv4ser.sin_port = htons(50000);      // 端口號轉網絡字節序(50000)ser.sin_addr.s_addr = inet_addr("192.168.1.48");  // 服務器IP地址// 打開本地文件(只讀模式)int fd = open("/home/linux/1.png", O_RDONLY);if (-1 == fd){perror("open error\n");return 1;}int num = 0;        // 累計已發送字節數char buf[1024] = {0};  // 數據緩沖區(1024字節塊)while (1){bzero(buf, sizeof(buf));  // 清空緩沖區int ret = read(fd, buf, sizeof(buf));  // 從文件讀取數據到緩沖區num += ret;  // 累計傳輸字節數printf("num:%d\n", num);  // 實時打印已發送字節數// 讀取結束或出錯時退出循環if (ret <= 0){break;}// 向服務器發送數據包sendto(sockfd, buf, ret, 0, (SA)&ser, sizeof(ser));// 接收服務器確認包(小阻塞:確保服務器處理完當前包)bzero(buf, sizeof(buf));recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL);}// 發送傳輸結束標志strcpy(buf, "^_^");  // 結束標志字符串sendto(sockfd, buf, strlen(buf), 0, (SA)&ser, sizeof(ser));// 關閉資源close(sockfd);close(fd);return 0;
}

服務器代碼 (ser.c)

#include <arpa/inet.h>   // 提供IP地址轉換函數
#include <netinet/in.h>  // 定義IPv4地址結構
#include <netinet/ip.h>  // 定義IP協議相關結構
#include <stdio.h>       // 標準輸入輸出函數
#include <stdlib.h>      // 標準庫函數
#include <string.h>      // 字符串操作函數
#include <sys/socket.h>  // 套接字API
#include <sys/types.h>   // 數據類型定義
#include <unistd.h>      // POSIX系統調用
#include <time.h>        // 時間相關函數
#include <fcntl.h>       // 提供文件控制操作typedef struct sockaddr * (SA);  // 定義sockaddr指針別名int main(int argc, char **argv)
{// 創建UDP套接字int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (-1 == sockfd){perror("socket");return 1;}// 配置服務器地址結構(用于綁定)struct sockaddr_in ser, cli;ser.sin_family = AF_INET;         // 地址族:IPv4ser.sin_port = htons(50000);      // 端口號轉網絡字節序ser.sin_addr.s_addr = inet_addr("192.168.1.48");  // 本機IP地址// 綁定套接字到指定地址和端口int ret = bind(sockfd, (SA) &ser, sizeof(ser));if (-1 == ret){perror("bind");return 1;}time_t tm;  // 時間變量(未實際使用,保留原代碼)socklen_t len = sizeof(cli);  // 客戶端地址結構長度// 創建新文件(寫入/創建/截斷模式,權限0666)int fd = open("2.png", O_WRONLY | O_CREAT | O_TRUNC, 0666);if (-1 == fd){perror("open error\n");return 1;}int num = 0;  // 累計接收字節數while (1){char buf[1024] = {0};  // 數據緩沖區// 接收客戶端數據包int ret = recvfrom(sockfd, buf, sizeof(buf), 0, (SA)&cli, &len);num += ret;  // 累計接收字節數printf("num:%d\n", num);  // 實時打印已接收字節數// 檢測到結束標志時退出循環if (0 == strcmp(buf, "^_^")){break;}// 將數據寫入文件write(fd, buf, ret);// 發送確認包(解除客戶端阻塞)bzero(buf, sizeof(buf));strcpy(buf, "go on");  // 確認消息sendto(sockfd, buf, strlen(buf), 0, (SA)&cli, len);}// 關閉資源close(sockfd);close(fd);return 0;
}

理想運行結果

  1. 文件傳輸過程
    • 客戶端輸出:num:1024num:2048 → … → num:[文件總大小](字節數遞增)
    • 服務器輸出:num:1024num:2048 → … → num:[文件總大小](與客戶端一致)
  2. 傳輸驗證
    • 傳輸完成后,2.png 與原始 1.png 大小完全相同(ls -l 對比)
    • 圖片文件可正常打開(如 eog 2.png 無損壞)
  3. 退出機制
    • 客戶端發送 ^_^ 后自動退出
    • 服務器收到 ^_^ 后自動退出
  4. 關鍵特性
    • 每個數據包傳輸后通過 recvfrom/sendto 實現簡單確認,避免發送過快導致丟包
    • 結束標志 ^_^ 確保雙方同步退出

UDP聊天

實現雙向實時聊天功能,支持連續消息收發。輸入 #quit 時,客戶端和服務器同時退出。核心知識點包括:多進程并發處理(fork)、消息循環控制、退出信號同步。

服務器代碼 (ser.c)

#include <arpa/inet.h>   // IP地址轉換
#include <fcntl.h>       // 文件控制
#include <netinet/in.h>  // IPv4地址結構
#include <netinet/ip.h>  // IP協議結構
#include <signal.h>      // 信號處理(用于進程終止)
#include <stdio.h>       // 標準I/O
#include <stdlib.h>      // 標準庫
#include <string.h>      // 字符串操作
#include <sys/socket.h>  // 套接字API
#include <sys/types.h>   // 數據類型
#include <time.h>        // 時間函數
#include <unistd.h>      // POSIX系統調用typedef struct sockaddr *(SA);  // sockaddr指針別名int main(int argc, char **argv)
{// 創建UDP套接字int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (-1 == sockfd){perror("socket");return 1;}// 配置服務器地址結構struct sockaddr_in ser, cli;ser.sin_family = AF_INET;ser.sin_port = htons(50000);      // 監聽端口50000ser.sin_addr.s_addr = inet_addr("192.168.1.48");  // 本機IP// 綁定套接字int ret = bind(sockfd, (SA)&ser, sizeof(ser));if (-1 == ret){perror("bind");return 1;}socklen_t len = sizeof(cli);  // 客戶端地址長度char buf[512] = {0};          // 消息緩沖區// 等待客戶端初始連接("start"消息)recvfrom(sockfd, buf, sizeof(buf), 0, (SA)&cli, &len);// 創建子進程處理并發pid_t pid = fork();if (pid > 0)  // 父進程:接收客戶端消息{while (1){bzero(buf, sizeof(buf));  // 清空緩沖區// 接收客戶端消息recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL);// 檢測退出命令if (0 == strcmp(buf, "#quit\n")){kill(pid, 9);  // 終止子進程exit(1);       // 父進程退出}printf("cli:%s", buf);  // 顯示客戶端消息}}else if (0 == pid)  // 子進程:發送服務器消息{while (1){printf("to cli:");  // 提示輸入char buf[512] = {0};fgets(buf, sizeof(buf), stdin);  // 讀取用戶輸入// 發送消息到客戶端sendto(sockfd, buf, strlen(buf), 0, (SA)&cli, len);// 檢測退出命令if (0 == strcmp(buf, "#quit\n")){kill(getppid(), 9);  // 終止父進程exit(1);             // 子進程退出}}}else  // fork失敗{perror("fork");return 1;}close(sockfd);  // 關閉套接字(實際不會執行到此處)return 0;
}

客戶端代碼 (cli.c)

#include <arpa/inet.h>   // IP地址轉換
#include <netinet/in.h>  // IPv4地址結構
#include <netinet/ip.h>  // IP協議結構
#include <stdio.h>       // 標準I/O
#include <stdlib.h>      // 標準庫
#include <string.h>      // 字符串操作
#include <sys/socket.h>  // 套接字API
#include <sys/types.h>   // 數據類型
#include <unistd.h>      // POSIX系統調用
#include <time.h>        // 時間函數
#include <fcntl.h>       // 文件控制
#include <signal.h>      // 信號處理typedef struct sockaddr * (SA);  // sockaddr指針別名int main(int argc, char **argv)
{// 創建UDP套接字int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (-1 == sockfd){perror("socket");return 1;}// 配置服務器地址結構struct sockaddr_in ser;ser.sin_family = AF_INET;ser.sin_port = htons(50000);      // 目標端口50000ser.sin_addr.s_addr = inet_addr("192.168.1.48");  // 服務器IPsocklen_t len = sizeof(ser);  // 地址長度char buf[512] = {0};strcpy(buf, "start");  // 初始連接消息sendto(sockfd, buf, strlen(buf), 0, (SA)&ser, len);  // 通知服務器// 創建子進程處理并發pid_t pid = fork();if (pid > 0)  // 父進程:接收服務器消息{while (1){bzero(buf, sizeof(buf));// 接收服務器消息recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL);// 檢測退出命令if (0 == strcmp(buf, "#quit\n")){kill(pid, 9);  // 終止子進程exit(1);       // 父進程退出}printf("ser:%s", buf);  // 顯示服務器消息}}else if (0 == pid)  // 子進程:發送客戶端消息{while (1){printf("to ser:");  // 提示輸入char buf[512] = {0};fgets(buf, sizeof(buf), stdin);  // 讀取用戶輸入// 發送消息到服務器sendto(sockfd, buf, strlen(buf), 0, (SA)&ser, len);// 檢測退出命令if (0 == strcmp(buf, "#quit\n")){kill(getppid(), 9);  // 終止父進程exit(1);             // 子進程退出}}}else  // fork失敗{perror("fork");return 1;}close(sockfd);  // 關閉套接字(實際不會執行到此處)return 0;
}

理想運行結果

  1. 初始化連接

    • 客戶端發送 start 消息,服務器開始監聽
    • 雙方進入消息循環
  2. 聊天過程

    • 服務器終端:
      to cli:Hello  // 服務器輸入
      cli:How are you?  // 顯示客戶端消息
      to cli:#quit  // 輸入退出命令
      
    • 客戶端終端:
      ser:Hello  // 顯示服務器消息
      to ser:How are you?  // 客戶端輸入
      ser:#quit  // 收到退出命令
      
  3. 退出機制

    • 任意一方輸入 #quit 后:
      • 發送方立即終止對方進程(kill
      • 本地進程退出(exit(1)
    • 雙方終端同時關閉,無殘留進程
  4. 關鍵特性

    • 通過 fork 實現雙工通信:父進程處理接收,子進程處理發送
    • 退出命令 #quit 觸發雙向終止(避免單方退出導致僵局)
    • 消息實時顯示(printf 前無緩沖,即時刷新)

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/95874.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/95874.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/95874.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

策略模式-不同的鴨子的案例

介紹了策略模式在C#中的應用&#xff0c;以一個鴨子的例子來說明。首先定義了鴨子類以及鴨子的行為&#xff08;方法&#xff09;&#xff0c;然后通過繼承和實現接口的方式來定義不同種類的鴨子的特性。介紹了策略模式的概念&#xff0c;將相同的算法封裝在不同的類中&#xf…

C++語言編程規范-初始化和類型轉換

01 C語言編程規范-常量 02 初始化和類型轉換 聲明、定義與初始化 03 禁止用 memcpy、memset 初始化非 POD 對象 說明&#xff1a;POD 全稱是“Plain Old Data”&#xff0c;是 C 98 標準(ISO/IEC 14882, first edition, 1998-09-01)中引入的一個概念&#xff0c; PO…

從零構建一款開源在線客服系統:我的Go語言實戰之旅

了解更多&#xff0c;搜索 "程序員老狼"用代碼連接世界&#xff0c;讓溝通無界限緣起&#xff1a;為什么選擇開發客服系統&#xff1f;在數字化浪潮席卷全球的今天&#xff0c;企業與客戶之間的溝通方式正在發生深刻變革。傳統的電話和郵件支持已無法滿足即時互動的需…

unsloth筆記:基本介紹

更快的速度、更省的內存訓練、運行、評估大模型 1 支持的模型 All Our Models | Unsloth Documentation 1.1 Dynamic GGUF/instruct 4-bit llama.cpp使用的新模型格式&#xff0c;專為高效、本地推理設計注&#xff1a;GGUF無法微調 只保留推理所需的內容&#xff0c;如量化…

博眾測控 | 一文看懂菊水電源產品在半導體測試中的應用

01 半導體在各行業上的應用半導體作為現代工業體系的“核心神經”&#xff0c;其性能參數與應用場景深度綁定&#xff0c;不同行業因核心設備的功能需求差異&#xff0c;對半導體的電流、電壓承載能力及類型選擇有著明確且嚴格的要求&#xff0c;具體應用細節如下&#xff1a;1…

【STM32】貪吃蛇 [階段 8] 嵌入式游戲引擎通用框架設計

這篇博客是 承接&#xff1a;【項目思維】貪吃蛇&#xff08;嵌入式進階方向&#xff09;中 嵌入式游戲引擎雛形&#xff08;終極進階&#xff09;&#xff0c; 是我們此前從 “寫一個小游戲”提升到“構建可復用游戲框架” 的飛躍階段。我們以“貪吃蛇游戲”為例&#xff0c;抽…

Vue圖標按鈕好用的樣式

圖標按鈕示例一 <template><div class"icon-button-group"><button class"icon-btn icon-btn--default"><i class"el-icon-moon"></i></button><button class"icon-btn icon-btn--primary"&g…

Nginx 實戰系列(一)—— Web 核心概念、HTTP/HTTPS協議 與 Nginx 安裝

文章目錄前言一、Web 概念1.1 Web 的基本概念1.1.1 Web的特點1.2 B/S 架構模型1.3 Web 請求與響應過程&#xff08;重點&#xff09;1.4 靜態資源與動態資源1.5 Web 的發展階段1.6 案例&#xff1a;搭建最小 Web 服務1.6.1 目標1.6.2 搭建步驟1.7 小結二、HTTP 與 HTTPS 協議2.…

一種用geoserver發布復雜樣式矢量服務的方法

最近因為系統需要在國產系統中部署&#xff0c;遇見了國產系統不支持ArcGIS的尷尬局面&#xff0c;好在geoserver還是可以支持的&#xff0c;遂用geoserver解決服務問題。 在發布過程中&#xff0c;遇到比較難受的點就是矢量數據的樣式配圖&#xff0c;在我用QGIS配好導出sld后…

為什么神經網絡網絡算法比機器學習模型算法更加強大?

神經網絡&#xff08;尤其是深度神經網絡&#xff09;相比傳統機器學習模型&#xff08;如線性回歸、決策樹、支持向量機等&#xff09;的“強大”主要體現在其更強的表達能力、自適應特征學習能力以及對復雜模式的建模能力。但這種“強大”并非絕對&#xff0c;而是有特定條件…

中國移動浪潮云電腦CD1000-系統全分區備份包-可瑞芯微工具刷機-可救磚

中國移動浪潮云電腦CD1000-系統全分區備份包-可瑞芯微工具刷機-可救磚 開啟ADB教程&#xff1a; 可查看&#xff1a;浪潮CD1000-移動云電腦-RK3528芯片-232G-安卓9-開啟ADB ROOT破解教程 可輕松打開了wifi adb和USB調試。 往期詳細內容-文章&#xff1a;浪潮CD1000-移動云電腦…

C++兩個字符串的結合

這段代碼實現字符串拼接功能。用戶輸入兩個字符串a和b后&#xff0c;使用append()方法將b追加到a后面&#xff0c;然后輸出拼接后的結果。代碼簡潔但存在改進空間&#xff1a;1. 缺少輸入驗證 2. 直接修改原字符串a可能不符合某些場景需求 3. 可考慮更高效的拼接方式。適合基礎…

UE4 Rider調試時添加自定義命令行參數

1、打開 Rider 右上角&#xff0c;針對你的項目&#xff08;例如叫做“Mini”&#xff09;打開 Edit 2、輸入自定義的參數&#xff0c;如下圖的例子是輸入 -dx12 -norhithread &#xff0c;然后Apply并OK。3、開始調試&#xff08;蟲子按鈕&#xff09;

混合架構大型語言模型(Jamba)

Jamba是由AI21 Labs開發的混合架構大型語言模型&#xff08;LLM&#xff09;&#xff0c;結合了Transformer的語義理解能力和Mamba結構化狀態空間模型&#xff08;SSM&#xff09;的高效性&#xff0c;旨在解決長文本處理中的計算瓶頸。 一、技術特點 1.混合架構設計 Jamba采用…

2025 年高教社杯全國大學生數學建模競賽C 題 NIPT 的時點選擇與胎兒的異常判定詳解(一)

基于胎兒Y染色體濃度的孕周與BMI建模分析摘要本文利用某競賽提供的胎兒Y染色體濃度數據&#xff0c;建立了以孕周和孕婦BMI為自變量的多項式回歸模型&#xff0c;探討了其對Y染色體濃度的影響。通過數據清洗與篩選&#xff0c;共獲得1082條有效男胎樣本。結果顯示&#xff1a;Y…

PyTorch DDP 隨機卡死復盤:最后一個 batch 掛起,NCCL 等待不返回

PyTorch DDP 隨機卡死復盤&#xff1a;最后一個 batch 掛起&#xff0c;NCCL 等待不返回&#xff0c;三步修復 Sampler & drop_last很多人在接觸深度學習的過程往往都是從自己的筆記本開始的&#xff0c;但是從接觸工作后&#xff0c;更多的是通過分布式的訓練來模型。由于…

計算機專業考研備考建議

對于全國碩士研究生招生考試&#xff08;考研&#xff09;&#xff0c;考試科目主要由兩大部分組成&#xff1a;全國統一命題的公共課 和 由招生單位自主命題的專業課。具體的考試科目取決于你報考的專業和學校。下面我為你詳細拆解&#xff1a;一、考試科目構成&#xff08;絕…

關于嵌入式學習——單片機1

基礎整體概念以應用為中心:消費電子(手機、藍牙耳機、智能音響)、醫療電子(心率脈搏、呼吸機)、無人機(大疆D)、機器人(人形四足機器人) 計算機技術:計算機五大組成:運算器(數據運算)、控制器(指令控制)、存儲器(內存外存)、輸入設備(鼠標、鍵盤、攝像頭)、輸出設備(顯示器)軟件…

LightDock.server liunx 雙跑比較

LightDock: a new multi-scale approach to protein–protein docking The LightDock server is free and open to all users and there is no login requirement server 1示例 故去除約束 next step 結果有正有負合理 2.常見警告? Structure contains HETATM entries. P…

SQL面試題及詳細答案150道(61-80) --- 多表連接查詢篇

《前后端面試題》專欄集合了前后端各個知識模塊的面試題,包括html,javascript,css,vue,react,java,Openlayers,leaflet,cesium,mapboxGL,threejs,nodejs,mangoDB,MySQL,Linux… 。 前后端面試題-專欄總目錄 文章目錄 一、本文面試題目錄 61. 什么是內連接(INNE…