linux线程的创建、退出、等待、取消、分离

进程与线程

  • 进程是计算机拥有资源的基本单位;线程是计算机调度的基本单位,是CPU分配时间片的基本单元
  • 一个进程可以拥有>=1个线程,同一进程的多个线程共享地址空间,文件描述符,每种信号的处理方式,当前工作目录,用户ID和组ID。每个线程私有线程号,寄存器(程序计数器,栈指针),堆栈,信号屏蔽字,调度优先级,线程私有的存储空间。
  • 线程引入原因:
    • 创建属于同一进程的另一线程比创建另一个进程花费小很多。
    • 同一进程内线程切换速度快。
    • 线程通信更加简单,不需要通过操作系统。

API

线程创建
#include <pthread.h>
int pthread_create(pthread_t * thread,const pthread_attr_t * attr,
                    void *(*start_routine)(void *),void * arg);
参数 意义
thread 指向线程标识符的指针,用来存储线程标识符
attr 设置线程属性,主要设置与栈相关的属性,一般情况下该参数设置为NULL,新的线程将使用系统默认的属性
start_routine 线程运行函数的起始地址,即在此线程中运行哪些代码
arg 运行函数的参数地址

返回值:成功:0,错误:出错编号。
pthread不是Linux系统默认的库而是POSIX线程库。在Linux中将其作为一个库来使用,因此编译时需要加上-pthread以显式链接该库

获取线程当前ID
include <pthread.h>
pthread_t pthread_self(void);

返回线程ID
线程标识符在进程中是唯一的,即分别属于两不同进程的两个线程可能有相同的线程标识符

#include <pthread.h>
#include <unistd.h>
#include <iostream>
using namespace std;

void * thr_fun(void * arg)
{

    cout << "Im thread" << endl;
    return((void*)0);
}

int main()
{

    int ret;
    pthread_t ntid;
    
    ret = pthread_create(&ntid,NULL,thr_fun,NULL);
    if(ret != 0)
    {
        perror("pthreadcreate");
        exit(1);
    }
    sleep(2);   //attention
    exit(0);
}
线程退出与等待
  • 线程退出
    线程主动结束
#include <pthread.h>
void pthread_exit(void * retval);

retval:返回信息

主线程用pthread_exit还是return
用pthread_exit只会使主线程自身退出,产生的子线程继续执行;用return则所有线程退出。

  • 线程等待
#include <pthread.h>
int pthread_join(pthread_t thread,void **retval);

参数表:
thread: 要等待的线程的pid
retval:用来存储被等待线程的返回值
返回0:成功;返回错误号:失败
主线程阻塞自己,等待子线程结束,然后回收子线程资源

示例代码:
#include <pthread.h>
#include <unistd.h>
#include <iostream>
using namespace std;

void * thr_fun(void * arg)
{

    int i = 5;
    while(i>0)
    {
        sleep(1);
        cout << "thread " <<pthread_self()<<" is running"<< endl;
        i--;
    }
    pthread_exit((void *)"testtest");//返回出错信息
}

int main()
{

    int ret;
    void * thread1_ret;
    pthread_t thread1;
    ret = pthread_create(&thread1,NULL,thr_fun,NULL);
    if(ret != 0)
    {
        perror("pthreadcreate");
        exit(1);
    }
    
    pthread_join(thread1,&thread1_ret);//获取出错信息
    cout <<(char*)thread1_ret << endl;
    exit(0);
}

线程终止
#include <pthread.h>
int pthread_cancel(pthread_t thread);
示例代码
#include <pthread.h>
#include <unistd.h>
#include <iostream>
using namespace std;

void * thr_fun(void * arg)
{

    cout << "thread " <<pthread_self()<<" will running for 5 seconds"<< endl;
    int i = 5;
    while(i>0)
    {
        sleep(1);
        cout << "thread " <<pthread_self()<<" is running"<< endl;
        i--;
    }
    return((void*)0);
}

int main()
{

    int ret;
    pthread_t thread1;

    ret = pthread_create(&thread1,NULL,thr_fun,NULL);
    if(ret != 0)
    {
        perror("pthreadcreate");
        exit(1);
    }

    sleep(4);
    cout << "cancel thread " << thread1 << "!" << endl;

    ret = pthread_cancel(thread1);
    if(ret != 0)
    {
        perror("pthreadcancel");
        exit(1);
    }
    ret = pthread_join(thread1,NULL);
    if(ret != 0)
    {
        perror("pthreadJOIN");
        exit(1);
    }
    exit(0);
}
取消状态设置

可以设置线程能否被取消和取消后是否立即执行

  • 取消状态设置
#include <pthread.h>
int pthread_setcancelstate(int state,int * oldstate);

参数表
state:PTHREAD_CANCEL_DISABLE或者PTHREAD_CANCEL_ENABLE
oldstate:指针类型,上一次取消状态的指针,可设NULL

  • 取消类型设置(取消是否立即生效)
#include <pthread.h>
int pthread_setcanseltype(int type,int old *type)

type:PTHREAD_CANCEL_ASYNCHRONOUS立即取消
PTHREAD_CANCEL_DEFERRED等待事件(如pthread_join时)才取消

线程分离

在任何一个时间点上,线程是可结合的(joinable),或者是分离的(detached)。一个可结合的线程能够被其他线程收回其资源和杀死,只有当pthread_join()函数返回时,创建的线程才算终止,才能释放自己占用的系统资源;在被其他线程回收之前,它的存储器资源(如栈)是不释放的。相反,一个分离的线程是不能被其他线程回收或杀死的,它的存储器资源在它终止时由系统自动释放。因此为了避免内存泄漏,所有线程的终止,要么已设为DETACHED,要么就需要使用pthread_join()来回收

#include <pthread.h>
int pthread_detach(pthread_t threadID);

返回0成功,错误号失败
分离后不可以再合并。该操作不可逆

综合以上要想让子线程总能完整执行(不会中途退出),

  • 一种方法是在主线程中调用pthread_join对其等待,即pthread_create/pthread_join/pthread_exit或return;
  • 一种方法是在主线程退出时使用pthread_exit,这样子线程能继续执行,即pthread_create/pthread_detach/pthread_exit;
  • 还有一种是pthread_create/pthread_detach/return,这时就要保证主线程不能退出,至少是子线程完成前不能退出。

注:很多地方参照了黄茹老师主编的《Linux环境高级程序设计》

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • 1.内存的页面置换算法 (1)最佳置换算法(OPT)(理想置换算法):从主存中移出永远不再需要的页面;如无这样的...
    杰伦哎呦哎呦阅读 3,237评论 1 9
  • 概述 线程和进程本质上来说都属于一个内核调度单元,也就是说都可以作为一条单独的执行路径。但是多进程程序通常有一些限...
    loopppp阅读 456评论 0 0
  • Linux/UNIX系统编程手册 [德] Michael Kerrisk 第29章 线程:介绍第30章 线程:线程...
    妖小灰阅读 596评论 0 0
  • 并发服务器 服务器分类 按连接类型分类 1.面向连接的服务器(如tcp) 2.面向无连接的服务器(如udp) 按处...
    小区保安阅读 1,079评论 0 1
  • (1) 进程与线程区别?进程是并发执行的程序在执行过程中分配和管理资源的基本单位,一个动态概概念,是竞争计算机资源...
    Lee_Lemon阅读 622评论 0 1