APUE學習之多線程編程(一):線程的創建和銷毀

来源:http://www.cnblogs.com/shenlinken/archive/2016/08/16/5775001.html
-Advertisement-
Play Games

一、線程標識 和每個進程都有一個進程ID一樣,每個線程也有一個線程ID,線程ID是以pthread_t數據類型來表示的,在Linux中,用無符號長整型表示pthread_t,Solaris 把phread_t數據類型表示為無符號整型,FreeBSD 和Mac OS X 用一個指向pthread結構的 ...


一、線程標識      和每個進程都有一個進程ID一樣,每個線程也有一個線程ID,線程ID是以pthread_t數據類型來表示的,在Linux中,用無符號長整型表示pthread_t,Solaris 把phread_t數據類型表示為無符號整型,FreeBSD 和Mac OS X 用一個指向pthread結構的指針來表示pthread_t數據類型。      可以使用pthread_self函數獲得自身的線程ID。   
#include <pthread.h>
 pthread_t pthread_self(void);
  二、線程創建      使用pthread_create函數創建新線程 
#include <pthread.h>
int pthread_create(pthread_t *restrict tidp, const pthread_arrt_t *restrict addr, void *(*start_rtn)(void *), void *restrict arg);
     當pthread_create成功返回後,新創建線程的線程ID會被設置成tidp指向的記憶體單元,attr參數用於定製各種不同的線程屬性,後面再討論線程屬性,現在先把它置為null,創建一個具有預設屬性的線程。      新創建的線程從start_rtn函數開始運行,該函數接收一個無類型指針的參數arg,如果要傳給它的參數多於一個,可以把參數放到一個結構中,然後把結構的地址作為arg傳入。      線程新建後會繼承調用線程的浮點環境和屏蔽字。 例子:  
#include "apue.h"
#include <pthread.h>

pthread_t ntid;

void printids(const char *s)
{
    pid_t pid;
    pthread_t tid;

    pid = getpid();
    tid = pthread_self();
    printf("%s pid %lu tid %lu (0x%lx)\n", s, (unsigned long)pid, (unsigned long)tid, (unsigned long)tid);
}

void *thr_fn(void *arg)
{
    printids("new thread: ");
    return ((void *)0);
}

int main(void)
{
    int err;

    err = pthread_create(&ntid, NULL, thr_fn, NULL);
    if (err != 0)
    {
        err_exit(err, "can't create thread");
    }

    printids("main thread: ");
    sleep(1);
    exit(0);
}
View Code   這個程式有兩個特別的地方:第一,主線程需要休眠,如果主線程不休眠,主線程會退出,新線程並沒有機會運行。第二,新線程通過pthread_self(),獲得自己的線程ID。
./a.out
main thread:  pid 27335 tid 3076404928 (0xb75e36c0)
new thread:  pid 27335 tid 3076401984 (0xb75e2b40)
View Code   雖然Linux線程ID是用無符號長整型來表示的,但它們看起來更像指針。   三、線程終止      如果任意線程調用了exit,_exit,_Exit,整個進程都會終止,這個要註意。      單個線程可以通過以下三種方式退出,且不終止整個進程。      1.線程可以簡單地從啟動常式中返回,返回值是線程的退出碼。      2.線程可以被同一進程中的其他線程取消。      3.調用pthread_exit        先來看pthread_exit退出的情況。
#include <pthread.h>
void pthread_exit(void *rval_ptr);

     ravl_ptr是無類型指針,進程中的其他線程可以通過pthread_join函數獲得這個指針。

#include <pthread.h>
int pthread_join(pthread_t thread, void **rval_ptr);

     調用線程將一直阻塞,直至指定的線程退出,rval_ptr就包含返回碼,如果線程被取消,rval_ptr指定的記憶體單元就設置為PTHREAD_CANCELED.可以通過調用pthread_join自動把線程置於分離狀態,如果線程已處於分離狀態,pthread_join就會調用失敗。

例子:
#include "apue.h"
#include <pthread.h>
 
void *thr_fn1(void *arg)
{
    printf("thread 1 returning\n");
    return (void *)1;
}
 
void *thr_fn2(void *arg)
{
    printf("thread 2 exiting\n");
    pthread_exit((void *)2);
}
 
int main(void)
{
    int err;
    pthread_t tid1, tid2;
    void *tret;
 
    err = pthread_create(&tid1, NULL, thr_fn1, NULL);
 
    if (err != 0)
    {
        err_exit(err, "can't create thread1");
    }
 
    err = pthread_create(&tid2, NULL, thr_fn2, NULL);
 
    if (err != 0)
    {
        err_exit(err, "can't create thread2");
    }
 
    err = pthread_join(tid1, &tret);
 
    if (err != 0)
    {
        err_exit(err, "can't join thread1");
    }
 
    printf("thread1 exit code:%ld\n", (long)tret);
 
    err = pthread_join(tid2, &tret);
 
    if (err != 0)
    {
        err_exit(err, "can't join thread2");
    }
 
    printf("thread2 exit code:%ld\n", (long)tret);
 
    return 0;
}
View Code
./a.out
thread 2 exiting
thread 1 returning
thread1 exit code:1
thread2 exit code:2
View Code   也可傳遞包含複雜消息的結構的地址,不過必須註意,這個結構所使用的記憶體必須在完成調用後仍是有效的。       線程也可以調用pthread_cancel函數來請求取消同一進程的其他線程
#include <pthread.h>
int pthread_cancel(pthread_t tid);

     聽著有點霸道,不過也只是請求而已,線程可以選擇忽略這個請求。

     線程可以安排它退出時需要調用的函數,這樣的函數是由pthread_cleanup_push註冊在棧中的,所以執行順序與註冊時相反。
#include <pthread.h>
void pthread_cleanup_push(void(*rtn)(void *), void *arg);
void pthread_cleanup_pop(int execute);

     當線程執行以下動作時,清理函數rtn由pthread_cleanup_push函數調度

     1.調用pthread_exit時      2.響應取消請求時      3.用非零execute參數調用pthread_cleanup_pop時。 例子:
#include "apue.h"
#include <pthread.h>
 
void cleanup(void *arg)
{
    printf("cleanup: %s\n", (char *)arg);
}
 
void *thr_fn1(void *arg)
{
    printf("thread 1 start\n");
    pthread_cleanup_push(cleanup, "thread 1 first handler");
    pthread_cleanup_push(cleanup, "thread 1 second handler");
    printf("thread 1 push complete\n");
 
    if (arg)
    {
        return (void *)1;
    }
 
    pthread_cleanup_pop(0);
    pthread_cleanup_pop(0);
 
    return (void *)1;
}
 
void *thr_fn2(void *arg)
{
    printf("thread 2 start\n");
    pthread_cleanup_push(cleanup, "thread 2 first handler");
    pthread_cleanup_push(cleanup, "thread 2 second handler");
    printf("thread 2 push complete\n");
 
    if (arg)
    {
        pthread_exit((void *)2);
    }
 
    pthread_cleanup_pop(0);
    pthread_cleanup_pop(0);
 
    return (void *)2;
}
 
int main(void)
{
    int err;
    pthread_t tid1, tid2;
    void *tret;
 
    err = pthread_create(&tid1, NULL, thr_fn1, (void *)1);
    if (err != 0)
    {
        err_exit(err, "can't create thread 1");
    }
 
    err = pthread_create(&tid2, NULL, thr_fn2, (void *)1);
    if (err != 0)
    {
        err_exit(err, "can't create thread 2");
    }
 
    err = pthread_join(tid1, &tret);
    if (err != 0)
    {
        err_exit(err, "can't join with thread 1");
    }
    printf("thread 1 exit code %ld\n", (long)tret);
 
    err = pthread_join(tid2, &tret);
    if (err != 0)
    {
        err_exit(err, "can't join with thread 2");
    }
    printf("thread 2 exit code %ld\n", (long)tret);
 
    return 0;
}
View Code 
./a.out
thread 2 start
thread 2 push complete
cleanup: thread 2 second handler
cleanup: thread 2 first handler
thread 1 start
thread 1 push complete
thread 1 exit code 1
thread 2 exit code 2
View Code      可知如果線程是通過它的啟動常式中返回而終止的話,它的清理處理程式就不會被調用。        在預設情況下,線程的終止狀態會一直保存到對該線程調用pthread_join,如果該線程已經被分離,則底層的資源可以線上程終止時立即被收回,不用再調用pthread_join,且再調用pthread_join會出錯。
#include <pthread.h>
int pthread_detach(pthread_t tid);

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 包(package)是多個函數的集合,常作為分享代碼的基本單元,代碼封裝成包可以方便其他用戶使用。越來越多的R包正在由世界上不同的人所創建並分發,這些分發的R包,可以從CRAN 或 github 上獲取,由於向 CRAN 提交包審核非常嚴格,有些開發者並沒有將自己開發的R包提至CRAN的意向,通過 ...
  • 參考文章:微信公眾號文章 在sql中怎麼查看一個字元的ascii編碼,so easy !! 結果是這樣的 還有一點,Unicode編碼只編碼第一個 ...
  • 黎俊傑 | 2016-07-28 14:37 聲明:部分表名為了脫敏而用XX代替 1、故障現象 (1)一個業務系統輸入用戶名與密碼後無法進入首頁,表現為一直在運行等待,運行緩慢 (2)整個系統無法正常使用,接近停運狀態 2、故障解決方法 調整資料庫參數alter system setevent='1... ...
  • 原創 2016-07-05 熊軍 Oracle 編輯手記:在理解Oracle技術細節時,我們不僅應該讀懂概念,還要能夠通過測試驗證細節,理解那些『功夫在詩外』的部分,例如全表掃描和單塊讀。 開發人員在進行新系統上線前的數據校驗測試時,發現一條手工執行的 SQL 執行了超過1小時還沒有返回結果。SQL... ...
  • 跟蹤標記:3604 功能: 輸出DBCC命令返回結果到查詢視窗(通常是SSMS視窗),類似print命令的顯示效果; 用途: 常用於獲取DBCC IND, DBCC PAGE命令的輸出結果,因為這2個命令預設不輸出任何結果。 舉例: 跟蹤標記:3605 功能: 輸出DBCC命令返回結果到SQL Se ...
  • 頁併發訪問的保護:閂鎖 在多線程併發情況下,需要防止讀線程讀到寫線程正在寫的資源,在編程中,通過使用互斥器(Mutexes), 信號量(Semaphore), 臨界區(Critical Section)和事件(Event Object)來保護資源,而這些機制在SQL Server中被統一表示為 閂鎖 ...
  • http://blog.csdn.net/cyuyan112233/article/details/50190927 http://beikeit.com/post-514.html ...
  • 在 PE文件頭的 IMAGE_OPTIONAL_HEADER 結構中的 DataDirectory(數據目錄表) 的第二個成員就是指向輸入表的。每個被鏈接進來的 DLL文件都分別對應一個 IMAGE_IMPORT_DESCRIPTOR (簡稱IID) 數組結構。 typedef struct _IM... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...