精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Linux開發_采用線程處理網絡請求

DS小龍哥-嵌入式技術 ? 2022-09-17 15:21 ? 次閱讀

【摘要】 介紹Linux網絡編程、線程編程,select機制,利用子線程響應TCP服務器的請求。

??TCP服務器創建流程:

1.?創建套接字

2.?綁定IP地址和端口號(創建服務器)

3.?設置監聽的數量(限制最大可以連接的客戶端數量)

4.?等待客戶端連接

5.?實現基本通信

??TCP客戶端創建流程

1.?創建套接字

2.?連接服務器

3.?實現基本通信

任務1:網絡編程

練習:

1.?實現TCP服務器與TCP客戶端之間的基本通信,收發數據 (按照上課的思路流程看函數文檔)

2.?實現TCP服務器與TCP客戶端之間的文件傳輸。(單個文件傳輸)

驗證方式:(1) 同一臺電腦演示 (2)同桌之間演示

考慮的問題:

(1)?網絡的傳輸環境,考慮應答問題

(2)?數據丟包之后如何處理? 可以重發

(3)?超時處理

(4)?服務器與客戶端之間連接斷開處理。(客戶端和服務器兩邊都需要重新連接)

文件傳輸可以在廣告機中使用。

(擴展要求): 顯示接收進度百分比,顯示接收的文件名稱,推薦: 定義結構體(使用數據結構)

3.?(擴展)實現TCP服務器與TCP客戶端之間的目錄傳輸。

4.?(擴展)實現網絡聊天室(模仿QQ發送消息的效果)

??一般情況下,推薦最大每次傳輸的字節數不超過1024字節。

任務2:線程編程

#include

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,

void *(*start_routine) (void *), void *arg);

Compile and link with -pthread.

-lpthread

理解: 一個線程就是一個while(1)。

[root@wbyq linux-share-dir]# gcc app.c

/tmp/cccOs4TK.o: In function `main':

app.c:(.text+0x69): undefined reference to `pthread_create'

app.c:(.text+0x8d): undefined reference to `pthread_create'

collect2: ld 返回 1

[root@wbyq linux-share-dir]# gcc app.c -lpthread

練習:

1.?學習線程的基本使用

#include 
#include 
void *start_1(void *arg)
{
	while(1)
	{
		printf("123\n");
		sleep(1);
	}
}

void *start_2(void *arg)
{
	while(1)
	{
		printf("456\n");
		sleep(1);
	}
}

int main(int argc,char *argv[])
{
	pthread_t thread_1;
	pthread_t thread_2;
	pthread_create(&thread_1,NULL,start_1,NULL);
	pthread_create(&thread_2,NULL,start_2,NULL);
	while(1)
	{
		printf("789\n");
		sleep(1);
	}
	return 0;
}

2.?實現一個服務器實現多個客戶端的連接,實現通信。

思路: 一個客戶端就是一個獨立的線程。

3.?擴展練習: 實現服務器同時對多個客戶端進行文件發送。

(1)?服務器連接上一個客戶端就創建一個線程。

(2)?線程的函數需要寫幾個? 1個

1個函數需要考慮的問題: 函數的可重入性能!

需要考慮到資源搶占! 使用信號量!

(搶答器)

設置線程分離屬性:

#include 
#include 
char str1[]="123456";
char str2[]="abcdef";
void *start_1(void *arg)
{
	 printf("arg1=%s\n",arg);
	 sleep(1);
}

void *start_2(void *arg)
{
	printf("arg2=%s\n",arg);
	sleep(2);
}
int main(int argc,char *argv[])
{
	pthread_t thread_1;
	pthread_t thread_2;
	pthread_create(&thread_1,NULL,start_1,"線程1的參數傳遞測試");
	pthread_create(&thread_2,NULL,start_2,"線程2的參數傳遞測試");
	
	pthread_detach(thread_1); //設置線程的分離屬性
	pthread_detach(thread_2); //設置線程的分離屬性
	
	while(1)
	{
		
	}
	return 0;
}

任務3: select阻塞輪詢機制

(1)?服務器什么時候收到數據? read

(2)?客戶端什么時候收到數據? Read

(3)?客戶端如何判斷已經與服務器斷開連接?

使用select機制 ,當select函數返回值為1,read函數為0就表示斷開

(4)?服務器如何檢測客戶端已經斷開連接?

采用心跳包的模式: 規定客戶端每5秒鐘發送一個特定的數據給服務器。

??IO 多路復用是指內核一旦發現進程指定的一個或者多個 IO 條件準備讀取,它就通知該進程。

IO 條件:

(1)?網絡編程中的讀寫

(2)?標準輸出輸入中的讀寫

#include 

#include 

#include 

#include 
select:同時可以監控多個文件描述符。
int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout);

參數:

int nfds :最大的文件描述符+1

fd_set *readfds :讀事件發生

fd_set *writefds :寫事件發生

fd_set *exceptfds:出現問題

struct timeval *timeout:輪詢的時間。

填NULL表示無限阻塞。

結構體里的成員填0,表示不阻塞

結構體里的成員填>0,正常的阻塞時間

返回值: 0表示沒有任何事件發生,負數表示失敗。>0表示發生對應的事件。
void FD_CLR(int fd, fd_set *set); //清除指定文件描述符

int FD_ISSET(int fd, fd_set *set); //檢測指定的文件描述符是否發生了事件

void FD_SET(int fd, fd_set *set); //添加指定的文件描述符到fd描述符集合(多次調用)

void FD_ZERO(fd_set *set); //清除整個文件描述符集合
struct timeval {

long tv_sec; /* seconds */

long tv_usec; /* microseconds */

};

TCP服務器端處理:

#include 
#include           /* See NOTES */
#include 
#include    //使用大小端轉換函數
#include 
#include 
#include 
#include 
#include 
#include 
//函數聲明
void *start_routine_1(void *dev);
void *start_routine_2(void *dev);

typedef void *(*start_routine) (void *);
start_routine fun[]={start_routine_1,start_routine_2};
pthread_t thread_id[2];                //存放線程的標識符
int clientfd[2];                       //保存TCP客戶端的網絡套接字
struct sockaddr_in client_address[2];  //存放客戶端的信息
socklen_t address_len[2];              //存放客戶端結構體信息的長度
/*服務器端口號定義*/
#define P_host 8080

/*TCP服務器代碼*/
int main(int argc,char *argv[])
{
	 int socketfd;
	 struct sockaddr_in server_address;     //存放服務器的IP地址信息
	 
   memset(&server_address,0,sizeof(struct sockaddr_in)); //初始化內存空間
   memset(client_address,0,sizeof(struct sockaddr_in)*2); //初始化內存空間
   
	 server_address.sin_family=PF_INET;            //IPV4協議
	 server_address.sin_port=htons(P_host);        //端口號賦值
	 server_address.sin_addr.s_addr=INADDR_ANY;    //本地IP地址
  /*1 .創建套接字*/
   socketfd=socket(PF_INET,SOCK_STREAM,0);
   if(socketfd<0)
   	{
   	    printf("服務器網絡套接字創建失敗!\n");	
   	    return -1;
   	}
      	 
   /*2. 綁定端口,創建服務器*/
   if(bind(socketfd,(const struct sockaddr *)&server_address,sizeof(struct sockaddr))!=0)
   	{
   	    printf("服務器綁定端口失敗!\n");	
   	    return -1; 	
   	}
   
   /*3. 設監聽的端口數量*/
   if(listen(socketfd,10)!=0)
   	{
   		 printf("服務器端口監聽失敗!\n");	
   	   return -1;	 
   	}
   
    int i;
    
    for(i=0;i<2;i++)
   	{
   		 address_len[i]=sizeof(struct sockaddr);  //計算結構體大小 20 
   	  /*4. 等待客戶端連接*/
	    if((clientfd[i]=accept(socketfd,(struct sockaddr *)&client_address[i],&address_len[i]))<0)
	   	{
	   	   	printf("等待客戶端連接失敗!\n");	
	   	    break;	
	   	} 
	    
	    //創建線程
	   	if(pthread_create(&thread_id[i],NULL,fun[i],NULL)!=0)
			{
			   printf("線程_%d_創建失敗!\n",i);		
			} 	
   	}
   	
   	while(1)
   	{
   	    	
   	}
     	
   //阻塞方式等待線程的結束
	pthread_join(thread_id[0],NULL);
	pthread_join(thread_id[1],NULL);
	return 0;	
}
//線程1
void *start_routine_1(void *dev)
{
	 while(1)
	 {
	 	  printf("TCP客戶端1連接!\n");
	 	  sleep(2);
	 }
	 //終止線程
	 pthread_exit(NULL);
}
//線程2
void *start_routine_2(void *dev)
{
	while(1)
	{
	 	  printf("TCP客戶端2連接!\n");
	 	  sleep(2);
	}
	 //終止線程
	 pthread_exit(NULL);
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11225

    瀏覽量

    208911
  • 編程
    +關注

    關注

    88

    文章

    3591

    瀏覽量

    93592
  • 線程
    +關注

    關注

    0

    文章

    504

    瀏覽量

    19649
收藏 人收藏

    評論

    相關推薦

    socket 多線程編程實現方法

    在現代網絡編程中,多線程技術被廣泛應用于提高服務器的并發處理能力。Socket編程是網絡通信的基礎,而將多線程技術應用于Socket編程,可
    的頭像 發表于 11-12 14:16 ?223次閱讀

    Python編程:處理網絡請求的代理技術

    網絡編程中,代理技術扮演著至關重要的角色,尤其在處理網絡請求時。通過代理服務器,我們可以實現請求的轉發、緩存、負載均衡以及安全控制等功能。
    的頭像 發表于 11-12 07:23 ?133次閱讀

    Linux網絡協議棧的實現

    請求并與底層的網絡硬件進行交互。本文將深入探討 Linux 網絡協議棧的架構與實現,涵蓋數據包處理流程、關鍵模塊、協議棧層次以及性能優化等方
    的頭像 發表于 09-10 09:51 ?257次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>網絡</b>協議棧的實現

    嵌入式linux開發的基本步驟有哪些?

    之前,首先需要選擇合適的硬件平臺。硬件選擇需要考慮以下幾個方面: 1.1 處理器 嵌入式Linux開發需要一個處理器,常見的處理器有ARM、
    的頭像 發表于 09-02 09:11 ?397次閱讀

    嵌入式linux開發板芯片的工作原理

    處理器、存儲器、輸入/輸出接口等硬件資源的嵌入式系統開發平臺。它通常采用Linux操作系統作為底層軟件平臺,提供了豐富的開發工具和庫函數,
    的頭像 發表于 09-02 09:07 ?331次閱讀

    linux開發板與樹莓派的區別

    操作系統的微型計算機,主要用于教育、編程、媒體播放等領域。 硬件配置 Linux開發板:Linux開發板的硬件配置因廠商和型號而異,通常包括處理
    的頭像 發表于 08-30 15:34 ?757次閱讀

    linux開發板和單片機開發的區別

    硬件架構 Linux開發板和單片機開發在硬件架構上有很大的區別。Linux開發板通常基于ARM、x86或其他
    的頭像 發表于 08-30 15:30 ?628次閱讀

    鴻蒙OS開發實戰:【網絡管理HTTP數據請求

    應用通過HTTP發起一個數據請求,支持常見的GET、POST、OPTIONS、HEAD、PUT、DELETE、TRACE、CONNECT方法。
    的頭像 發表于 04-01 16:31 ?666次閱讀
    鴻蒙OS<b class='flag-5'>開發</b>實戰:【<b class='flag-5'>網絡</b>管理HTTP數據<b class='flag-5'>請求</b>】

    鴻蒙OS開發實例:【窺探網絡請求

    HarmonyOS 平臺中使用網絡請求,需要引入 "@ohos.net.http", 并且需要在 module.json5 文件中申請網絡權限, 即 “ohos.permission.INTERNET” 本篇文章將嘗試使用
    的頭像 發表于 04-01 16:11 ?726次閱讀
    鴻蒙OS<b class='flag-5'>開發</b>實例:【窺探<b class='flag-5'>網絡</b><b class='flag-5'>請求</b>】

    鴻蒙原生應用開發-網絡管理HTTP數據請求

    request請求先返回。可以根據業務需要訂閱此消息。 4.調用該對象的request()方法,傳入http請求的url地址和可選參數,發起網絡請求。 5.按照實際業務需要,解析返回結
    發表于 03-29 17:51

    鴻蒙開發實戰:網絡請求庫【axios】

    [Axios]?,是一個基于 promise 的網絡請求庫,可以運行 node.js 和瀏覽器中。本庫基于[Axios]原庫v1.3.4版本進行適配,使其可以運行在 OpenHarmony,并沿用其現有用法和特性。
    的頭像 發表于 03-25 16:47 ?3700次閱讀
    鴻蒙<b class='flag-5'>開發</b>實戰:<b class='flag-5'>網絡</b><b class='flag-5'>請求</b>庫【axios】

    請問RTOS USBSetuPcb回調函數是在中斷還是單獨線程處理的?

    UART 通信)和異步 USB Setup 請求回調處理程序函數之間的 Uart 訪問之間存在競爭條件。 -> 在執行主循環UART功能的同時,正在執行USB消息處理UART功能。 要
    發表于 02-23 06:10

    linux線程編程實例

    linux線程
    的頭像 發表于 02-15 21:16 ?415次閱讀
    <b class='flag-5'>linux</b>多<b class='flag-5'>線程</b>編程實例

    請問Spring Boot是如何快速過濾出一次請求所有日志的?

    在現網出現故障時,我們經常需要獲取一次請求流程里的所有日志進行定位 。如果請求只在一個線程處理,則我們可以通過線程ID來過濾 日志 ,但如
    的頭像 發表于 01-18 10:18 ?334次閱讀

    redis多線程還能保證線程安全嗎

    Redis是一種使用C語言編寫的高性能鍵值存儲系統,它是單線程的,因為使用了多路復用的方式來處理并發請求。這樣的實現方式帶來了很好的性能,但同時也引發了一些線程安全方面的問題。 在Re
    的頭像 發表于 12-05 10:28 ?1690次閱讀