主頁 > 後端開發 > Linux網路編程:socket & pthread_create()多執行緒 實作clients/server通信

Linux網路編程:socket & pthread_create()多執行緒 實作clients/server通信

2023-05-19 13:10:16 後端開發

一、問題引入

Linux網路編程:socket & fork()多行程 實作clients/server通信 隨筆介紹了通過fork()多行程實作了服務器與多客戶端通信,但除了多行程能實作之外,多執行緒也是一種實作方式,

重要的是,多行程和多執行緒是涉及作業系統層次,隨筆不僅要利用pthread_create()實作多執行緒編程,也要理解執行緒和行程的區別,

二、解決程序

client 代碼無需修改,請參考 Linux網路編程:socket & fork()多行程 實作clients/server通信

2-1 server 代碼

#include <stdlib.h>
#include <pthread.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>

#include <errno.h>
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <sys/syscall.h>

#define IP "10.8.198.227"
#define PORT 8887
#define gettid() syscall(__NR_gettid)
#define PTHREAD_MAX_SIZE 3    // 允許最大客戶端連接數

typedef struct PTHREAD_DATA_ST
{
    pthread_t pthread_id;
    int connfd;
    char socket[128];
    struct sockaddr_in cliaddr;
}PTHREAD_DATA_ST;

//static int g_pthread_num = 3; // 允許最大客戶端連接數
static struct PTHREAD_DATA_ST g_pthread_data[PTHREAD_MAX_SIZE];

static void pthread_data_index_init(void)
{
    for (int i = 0; i < PTHREAD_MAX_SIZE; i++)
    {
        memset(&g_pthread_data[i], 0 , sizeof(struct PTHREAD_DATA_ST));
        g_pthread_data[i].connfd = -1;
    }
}

static int pthread_data_index_find(void)
{
    int i;
    for (i = 0; i < PTHREAD_MAX_SIZE; i++)
    {
        if (g_pthread_data[i].connfd == -1)
            break;
    }
    return i;
}

static int string_toupper(const char *src, int str_len, char *dst)
{
    int count = 0;
    for (int i = 0; i < str_len; i++)
    {
        dst[i] = toupper(src[i]);
        count++;
    }
    return count;
}

void *pthread_handle(void *arg)
{
    struct PTHREAD_DATA_ST *pthread = (struct PTHREAD_DATA_ST *)arg;
    int connfd = pthread->connfd;
    int recv_len, send_len;
    pid_t tid = gettid();
    char read_buf[1024], write_buf[1024];
    while (1)
    {
        memset(read_buf, 0, sizeof(read_buf));
        memset(write_buf, 0, sizeof(write_buf));
        recv_len = read(connfd, read_buf, sizeof(read_buf));
        if (recv_len <= 0)
        {
            printf("%s close, child %d terminated\n", pthread->socket, tid);
            close(connfd);
            pthread->connfd = -1;
            pthread_exit(NULL);
        }
        printf("%s:%s(%d Byte)\n", pthread->socket, read_buf, recv_len);
        send_len = string_toupper(read_buf, strlen(read_buf), write_buf);
        write(connfd, write_buf, send_len);
        if (strcmp("exit", read_buf) == 0)
        {
            printf("%s exit, child %d terminated\n", pthread->socket, tid);
            close(connfd);
            pthread->connfd = -1;
            pthread_exit(NULL);
        }
    }
}

int main(void)
{
    int listenfd, connfd;
    struct sockaddr_in server_sockaddr;
    struct sockaddr_in client_addr;
    char buf[1024];
    char client_socket[128];
    socklen_t length;
    int idx;
    int opt = 1;

    server_sockaddr.sin_family = AF_INET;
    server_sockaddr.sin_port = htons(PORT);
    server_sockaddr.sin_addr.s_addr = inet_addr(IP);
    listenfd = socket(AF_INET, SOCK_STREAM, 0);
    if (listenfd < 0)
    {
        perror("socket error");
        exit(1);
    }
    // 設定埠復用
    setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)); 
    if (bind(listenfd, (struct sockaddr *)&server_sockaddr, sizeof(struct sockaddr)) < 0)
    {
        perror("bind error");
        exit(1);
    }
    if (listen(listenfd, 5) < 0)
    {
        perror("listen error");
        exit(1);
    }

    pthread_data_index_init();
    while (1)
    {
        // 接受來自客戶端的資訊
        printf("accept start \n");
        memset(&client_addr, 0, sizeof(client_addr));
        length = sizeof(client_addr);
        if ((connfd = accept(listenfd, (struct sockaddr *)&client_addr, &length)) < 0)
        {
            if (errno == EINTR)
                continue;
            else
            {
                perror("accept error");
                exit(1);
            }
        }

        idx = pthread_data_index_find();
        if (idx == PTHREAD_MAX_SIZE)
        {
            printf("client connected upper limit, refused connect\n");
            close(connfd);
            continue;
        }

        memset(&client_socket, 0, sizeof(client_socket));
        printf("client addr:%s por:%d\n",
               inet_ntop(AF_INET, &client_addr.sin_addr, buf, sizeof(buf)),
               ntohs(client_addr.sin_port));
        snprintf(client_socket, sizeof(client_socket), "client socket (%s:%d)",
                 inet_ntop(AF_INET, &client_addr.sin_addr, buf, sizeof(buf)),
                 ntohs(client_addr.sin_port));
        g_pthread_data[idx].connfd = connfd;
        g_pthread_data[idx].cliaddr = client_addr;
        strcpy(g_pthread_data[idx].socket, client_socket);
        pthread_create(&(g_pthread_data[idx].pthread_id), NULL, pthread_handle, &(g_pthread_data[idx]));
        pthread_detach(g_pthread_data[idx].pthread_id);
    }
    close(listenfd);
    return EXIT_SUCCESS;
}

2-2 編譯運行

?? 編譯server.c時,出現 對‘pthread_create’未定義的參考

原因:pthread.h庫不是linux系統的默認庫,添加編譯引數:-lpthread 即可 (或 pthread 也行)

  • 編譯源檔案

gcc server.c -g -std=gnu99 -lpthread -o server

  • client 1 和client 2連接server

  • client 1 斷開連接

  • client 2 斷開連接

?? 在server未設定埠復用,server主動斷開,重啟報錯: bind error: Adress already in use

解決方法:

int opt = 1;
// 設定埠復用
setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));

2-3 多執行緒處理

  • 執行緒id

?? 同一行程下的行程號pid相同,各個執行緒的執行緒號tid不相同

在linux中無法直接使用gettid()獲取執行緒id,正確解決辦法在源檔案中添加如下宏:

#include <sys/syscall.h>
#define gettid() syscall(__NR_gettid)
  • 多執行緒

通過宏 PTHREAD_MAX_SIZE 定義執行緒最大數量,畢竟計算機資源是有限的,

accept()成功回傳,此時某一個client已經完成TCP三次握手,接下來可以準備創建執行緒通信,但不能無腦創建執行緒,應檢測當前已創建的執行緒數量,只有未超越執行緒最大數量才能創建執行緒,

1、父執行緒負責監聽新的客戶端連接,并創建新的執行緒

2、子執行緒負責和客戶端進行通信

?? 注意:使用多執行緒要將子執行緒設定為分離屬性, 讓執行緒在退出之后自己回收資源

// pthread_create(), pthread_detach(), pthread_exit()原型宣告在如下頭檔案中
#include <pthread.h>

/*
 * @param *thread 新執行緒創建成功后的執行緒指標
 * @param *attr 創建執行緒時,設定屬性,默認可設定為NULL
 * @param *(*start_routine) (void *) 執行緒運行函式指標
 * @param *arg 執行緒運行函式指標的形參
 * @return 若成功,回傳值為0,否則失敗
 */
int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
                    void *(*start_routine) (void *), void *arg);

/* pthread_detact()在執行緒結束后,回收資源
 * @param thread 新執行緒創建成功后的執行緒指標
 * @return 若成功,回傳值為0,否則失敗
 */
int pthread_detach(pthread_t thread);

//Compile and link with -pthread.

執行緒呼叫 描述
pthread_create 創建一個新執行緒
pthread_exit 結束呼叫的執行緒
pthread_join 等待一個特定的執行緒退出
pthread_yield 釋放CPU來運行另外一個執行緒
pthread_attr_init 創建并初始化一個執行緒的屬性結構
pthread_attr_destroy 洗掉(銷毀)一個執行緒的屬性結構

三、反思總結

3-1 行程

在行程模型中,計算機上所有可運行的軟體,通常也包括草錯系統,被組織成若干順序行程,簡稱行程(process),

行程是資源(CPU、記憶體等)分配的基本單位,它是程式執行時的一個實體,

3-2 執行緒

執行緒是程式執行時的最小單位,它是行程的一個執行流,是CPU調度和分派的基本單位,一個行程可以由一個或多個執行緒組成,執行緒間共享行程的所有資源,每個執行緒有自己的堆疊和區域變數,

四、參考參考

UNIX網路編程 卷1:套接字聯網API 第3版

現代作業系統(第四版)

Unix網路編程學習筆記

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/552900.html

標籤:C

上一篇:驅動開發:通過應用堆實作多次通信

下一篇:返回列表

標籤雲
其他(159340) Python(38156) JavaScript(25439) Java(18070) C(15229) 區塊鏈(8267) C#(7972) AI(7469) 爪哇(7425) MySQL(7201) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5340) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4573) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2433) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1975) 功能(1967) Web開發(1951) HtmlCss(1940) python-3.x(1918) C++(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1878) .NETCore(1861) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Linux網路編程:socket & pthread_create()多執行緒 實作clients/s

    一、問題引入 Linux網路編程:socket & fork()多行程 實作clients/server通信 隨筆介紹了通過fork()多行程實作了服務器與多客戶端通信。但除了多行程能實作之外,多執行緒也是一種實作方式。 重要的是,多行程和多執行緒是涉及作業系統層次。隨筆不僅要利用pthread_cre ......

    uj5u.com 2023-05-19 13:10:16 more
  • 驅動開發:通過應用堆實作多次通信

    在前面的文章`《驅動開發:運用MDL映射實作多次通信》`LyShark教大家使用`MDL`的方式靈活的實作了內核態多次輸出結構體的效果,但是此種方法并不推薦大家使用原因很簡單首先內核空間比較寶貴,其次內核里面不能分配太大且每次傳出的結構體最大不能超過`1024`個,而最終這些記憶體由于無法得到更好的釋... ......

    uj5u.com 2023-05-19 12:58:14 more
  • 重慶高校平臺刷課軟體(小魚高校平臺助手)相關說明

    ###很久沒有更新博客了,最近忙著接一些js的腳本外包,忙著背各種面經八股文,今天把剛剛更新了的小魚高校平臺助手相關的一些東西說明一下吧 如圖目前掛在github下的軟體的官網被bing給收錄了,github內的原始碼地址也同樣被收錄其中,這讓我有一些驚喜 那么言歸正傳關于軟體出現的一些問題進行一下答 ......

    uj5u.com 2023-05-19 08:10:04 more
  • Fast-GRPC: 用python輕松開發grpc介面

    Fast-GRPC 框架旨在為開發者提供更輕松快捷的 Python gRPC 介面開發方式,受到 FastAPI 的啟發,用更human的方式撰寫 gRPC 介面,上手簡單、快速開發、支持 Middleware、支持同步異步 ......

    uj5u.com 2023-05-19 08:09:57 more
  • 反轉鏈表 Java版 圖文并茂思路分析帶答案(力扣第206題)

    反轉鏈表 力扣第206題 我們不只是簡單的學習(背誦)一個資料結構,而是要分析他的思路,以及為什么要有不同的指標等等 非遞回方式: 思路分析:首先要鏈表有個頭指標沒有任何問題 然后,我們要將1的下一個節點指向空,這樣才能將其反轉過來,但是這個時候我們發現和下一個節點2失去了聯系 所以我們要有一個指標 ......

    uj5u.com 2023-05-19 08:09:42 more
  • 民間最大社區,倒閉了!

    天涯神貼合集(500篇):https://pan.quark.cn/s/ba1e0577bfd8 最近幾天大家應該發現天涯社區網站打不開了。 天涯社區創辦于1999年,此時的中國,互聯網產業方興未艾,那時天涯社區相當火爆。 2007年時,天涯社區的注冊用戶就突破了2000萬,號稱是全球最大的中文互聯 ......

    uj5u.com 2023-05-19 08:09:32 more
  • Spring Boot整合Jwt

    JWT介紹 JWT是JSON Web Token的縮寫,即JSON Web令牌,是一種自包含令牌。 是為了在網路應用環境間傳遞宣告而執行的一種基于JSON的開放標準。 JWT的宣告一般被用來在身份提供者和服務提供者間傳遞被認證的用戶身份資訊,以便于從資源服務器獲取資源。比如用在用戶登錄上。 JWT最 ......

    uj5u.com 2023-05-19 08:09:27 more
  • java常用類

    java常用類 Object類 基類,超類,所有類的直接或間接父類 object類定義的方法是所有物件都具有的方法 object型別可以存盤任何物件 作為引數,可以接受任何物件 作為回傳值,可以回傳任何物件 getClass() 回傳參考中存盤的實際物件型別 public class Student ......

    uj5u.com 2023-05-19 08:09:09 more
  • python包管理工具:Conda和pip比較

    Conda和pip通常被認為幾乎完全相同。雖然這兩個工具的某些功能重疊,但它們設計用于不同的目的。 Pip是Python Packaging Authority推薦的用于從Python Package Index安裝包的工具。 Pip安裝打包為wheels或源代碼分發的Python軟體。后者可能要求 ......

    uj5u.com 2023-05-19 08:08:55 more
  • Java設計模式-外觀模式

    簡介 在軟體開發程序中,經常會遇到復雜的系統和龐大的類別庫。這些系統往往包含了大量的類和子系統,給開發人員帶來了挑戰。為了簡化介面設計和提高系統的可用性,設計模式提供了一種名為外觀模式的解決方案。 外觀模式是一種結構型設計模式,旨在為復雜系統提供一個簡化的介面。該模式通過隱藏底層系統的復雜性,提供一個 ......

    uj5u.com 2023-05-19 08:08:50 more