目录

  • 线程概要

    • Linux内核线程实现原理
    • 线程的共享/不共享资源
    • 线程优缺点
  • 线程控制原语
    • pthread_self
    • pthread_create
    • pthread_exit
    • pthread_join
    • pthread_cancel
    • 终止线程方式
    • 控制原语对比

前情提要: Linux用户级线程和内核级线程区别

线程概要

Linux内核线程实现原理

类Unix系统中,早期是没有“线程”概念的,80年代才引入,借助进程机制实现出了线程的概念。因此在这类系统中,进程和线程关系密切。

  1. 轻量级进程(light-weight process),也有PCB,创建线程使用的底层函数和进程一样,都是clone

  2. 从内核里看进程和线程是一样的,都有各自不同的PCB,但是PCB中指向内存资源的三级页表是相同的

  3. 进程可以蜕变成线程

  4. 线程可看做寄存器和栈的集合

  5. 在linux下,线程最是小的执行单位;进程是最小的分配资源单位

查看线程命令:ps -elf|grep thread

三级映射:进程PCB --> 页目录(可看成数组,首地址位于PCB中) --> 页表 --> 物理页面 --> 内存单元


线程的共享/不共享资源

线程共享资源 线程不共享资源
文件描述符表 线程id
每种信号的处理方式 处理器现场和栈指针(内核栈)
当前工作目录 独立的栈空间(用户空间栈)
用户ID和组ID errno变量
内存地址空间(.text/.data/.bss/heap/共享库) 信号屏蔽字
调度优先级

线程优缺点

优点: 1. 提高程序并发性 2. 开销小 3. 数据通信、共享数据方便

缺点: 1. 库函数,不稳定 2. 调试、编写困难、gdb不支持 3. 对信号支持不好

线程控制原语


pthread_self

获取线程ID。其作用对应进程中 getpid() 函数。

​ pthread_t pthread_self(void); 返回值:成功:0; 失败:无!

​ 线程ID:pthread_t类型,本质:在Linux下为无符号整数(%lu),其他系统中可能是结构体实现

​ 线程ID是进程内部,识别标志。(两个进程间,线程ID允许相同)


pthread_create

创建一个新线程。 其作用,对应进程中fork() 函数。

​ int pthread_create(pthread_t thread, const pthread_attr_t attr, void (start_routine) (void ), void arg);

​ 返回值:成功:0; 失败:错误号 -----Linux环境下,所有线程特点,失败均直接返回错误号。

参数:

​ pthread_t:当前Linux中可理解为:typedef unsigned long int pthread_t;

参数1:传出参数,保存系统为我们分配好的线程ID

​ 参数2:通常传NULL,表示使用线程默认属性。若想使用具体属性也可以修改该参数。

​ 参数3:函数指针,指向线程主函数(线程体),该函数运行结束,则线程结束。

​ 参数4:线程主函数执行期间所使用的参数。

练习:创建一个新线程,打印线程ID。注意:链接线程库 -lpthread

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>void *tfn(void *arg)
{printf("I'm thread, Thread_ID = %lu\n", pthread_self());return NULL;
}int main(void)
{pthread_t tid;pthread_create(&tid, NULL, tfn, NULL);sleep(1);printf("I am main, my pid = %d\n", getpid());return 0;
}

线程默认共享数据段、代码段等地址空间,常用的是全局变量,或者传参形式。而进程不共享全局变量,只能借助mmap。

全局变量:

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>
#include <unistd.h>int var = 100;void *tfn(void *arg)
{var = 200;printf("thread\n");return NULL;
}int main(void)
{printf("At first var = %d\n", var);pthread_t tid;pthread_create(&tid, NULL, tfn, NULL);sleep(1);printf("after pthread_create, var = %d\n", var);return 0;
}

传参:

#include <func.h>void *tfn(void *arg){int* var = (int*)arg;*var = 200;printf("thread\n");return NULL;
}int main()
{int var = 100;printf("At first var = %d\n", var);pthread_t tid;pthread_create(&tid, NULL, tfn, &var);sleep(1);printf("after pthread_create, var = %d\n", var);return 0;
}


pthread_exit

作用:将单个线程退出

​ void pthread_exit(void *retval); 参数:retval表示线程退出状态,通常传NULL

线程中,**禁止使用exit函数,会导致进程内所有线程全部退出**。所以,多线程环境中,应尽量少用,或者不使用exit函数,取而代之使用pthread_exit函数,将单个线程退出。任何线程里exit导致进程退出,其他线程未工作结束,主控线程退出时不能return或exit。

另注意,pthread_exit或者return返回的指针所指向的内存单元必须是全局的或者是用malloc分配的,不能在线程函数的栈上分配,因为当其它线程得到这个返回指针时线程函数已经退出了。


pthread_join

阻塞等待线程退出,获取线程退出状态 其作用,对应进程中 waitpid() 函数。

​ int pthread_join(pthread_t thread, void **retval); 成功:0;失败:错误号

​ 参数:thread:线程ID (【注意】:不是指针);retval:存储线程结束状态。

​ 对比记忆:

​ 进程中:main返回值、exit参数-->int;等待子进程结束 wait 函数参数-->int *

​ 线程中:线程主函数返回值、pthread_exit-->void *;等待线程结束 pthread_join 函数参数-->void **

调用该函数的线程将挂起等待,直到id为thread的线程终止。thread线程以不同的方法终止,通过pthread_join得到的终止状态是不同的,总结如下:

  1. 如果thread线程通过return返回,retval所指向的单元里存放的是thread线程函数的返回值。

  2. 如果thread线程被别的线程调用pthread_cancel异常终止掉,retval所指向的单元里存放的是常数PTHREAD_CANCELED。

  3. 如果thread线程是自己调用pthread_exit终止的,retval所指向的单元存放的是传给pthread_exit的参数。

  4. 如果对thread线程的终止状态不感兴趣,可以传NULL给retval参数。

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <stdlib.h>typedef struct {int a;int b;
} exit_t;void *tfn(void *arg)
{exit_t *ret;ret = malloc(sizeof(exit_t)); ret->a = 100;ret->b = 300;pthread_exit((void *)ret);
}int main(void)
{pthread_t tid;exit_t *retval;pthread_create(&tid, NULL, tfn, NULL);/*调用pthread_join可以获取线程的退出状态*/pthread_join(tid, (void **)&retval);      //wait(&status);printf("a = %d, b = %d \n", retval->a, retval->b);return 0;
}


pthread_cancel

杀死(取消)线程 其作用,对应进程中 kill() 函数。

​ int pthread_cancel(pthread_t thread); 成功:0;失败:错误号

​ 【注意】:线程的取消并不是实时的,而有一定的延时。需要等待线程到达某个取消点(检查点)。

​ 类似于玩游戏存档,必须到达指定的场所(存档点,如:客栈、仓库、城里等)才能存储进度。杀死线程也不是立刻就能完成,必须要到达取消点。

​ 取消点:是线程检查是否被取消,并按请求进行动作的一个位置。通常是一些系统调用creat,open,pause,close,read,write..... 执行命令man 7 pthreads可以查看具备这些取消点的系统调用列表。也可参阅 APUE.12.7 取消选项小节。

可粗略认为一个系统调用(进入内核)即为一个取消点。如线程中没有取消点,可以通过调用pthreestcancel函数自行设置一个取消点。

被取消的线程, 退出值定义在Linux的pthread库中。常数PTHREAD_CANCELED的值是-1。可在头文件pthread.h中找到它的定义:#define PTHREAD_CANCELED ((void *) -1)。因此当我们对一个已经被取消的线程使用pthread_join回收时,得到的返回值为-1。

终止线程的三种方法。注意“取消点”的概念。

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <stdlib.h>void *tfn1(void *arg)
{printf("thread 1 returning\n");return (void *)111;
}void *tfn2(void *arg)
{printf("thread 2 exiting\n");pthread_exit((void *)222);
}void *tfn3(void *arg)
{while (1) {//printf("thread 3: I'm going to die in 3 seconds ...\n");//sleep(1);pthread_testcancel();   //自己添加取消点*/}return (void *)666;
}int main(void)
{pthread_t tid;void *tret = NULL;pthread_create(&tid, NULL, tfn1, NULL);pthread_join(tid, &tret);printf("thread 1 exit code = %d\n\n", (int)tret);pthread_create(&tid, NULL, tfn2, NULL);pthread_join(tid, &tret);printf("thread 2 exit code = %d\n\n", (int)tret);pthread_create(&tid, NULL, tfn3, NULL);sleep(3);pthread_cancel(tid);pthread_join(tid, &tret);printf("thread 3 exit code = %d\n", (int)tret);return 0;
}

终止线程方式

总结:终止某个线程而不终止整个进程,有三种方法:

  1. 从线程主函数return。这种方法对主控线程不适用,从main函数return相当于调用exit。

  2. 一个线程可以调用pthread_cancel终止同一进程中的另一个线程。

  3. 线程可以调用pthread_exit终止自己。


控制原语对比

​ 进程 线程

​ fork pthread_create

​ exit pthread_exit

​ wait pthread_join

​ kill pthread_cancel

​ getpid pthread_self 命名空间

转载于:https://www.cnblogs.com/Mered1th/p/10801287.html

Linux系统编程——线程(1)相关推荐

  1. Linux系统编程——线程私有数据

    在多线程程序中.常常要用全局变量来实现多个函数间的数据共享.因为数据空间是共享的,因此全局变量也为全部线程共同拥有. 測试代码例如以下: #include <stdio.h> #inclu ...

  2. Linux系统编程——线程

    一.线程概念 基础 线程又称LWP:light weight process 轻量级的进程,(在linux环境下)本质仍是进程.进程:独立地址空间,拥有PCB 线程:有独立的PCB,但没有独立的地址空 ...

  3. Linux系统编程——线程池

    http://blog.csdn.net/tennysonsky/article/details/46490099# 线程池基本原理 在传统服务器结构中,常是有一个总的监听线程监听有没有新的用户连接服 ...

  4. 入门Linux系统编程--网络编程

    文章目录 一.网络编程 1.socket服务端代码实现(无连接客户端) 6.socket服务端代码实现(连接客户端) 7.socket客户端代码实现 8.实现双方聊天 9.多方消息收发 二.往期文章 ...

  5. Linux系统编程之进程与线程控制原语对比

    Linux系统编程之进程与线程控制原语对比 进程 线程 fork pthread_create exit pthread_exit wait pthread_join kill pthread_can ...

  6. linux线程并不真正并行,Linux系统编程学习札记(十二)线程1

    Linux系统编程学习笔记(十二)线程1 线程1: 线程和进程类似,但是线程之间能够共享更多的信息.一个进程中的所有线程可以共享进程文件描述符和内存. 有了多线程控制,我们可以把我们的程序设计成为在一 ...

  7. Linux系统编程(九)线程同步

    Linux系统编程(九)线程同步 一.什么是线程同步? 二.互斥量 三.条件变量 pthread_cond_wait函数 pthread_cond_signal函数 生产者和消费者模型 一.什么是线程 ...

  8. Linux系统编程(八)线程

    Linux系统编程(八)线程 一.什么是线程? 二.Linux内核线程实现原理 线程共享资源 线程非共享资源 线程优缺点 线程控制原语 一.什么是线程? LWP:light weight proces ...

  9. 【Linux | 系统编程】Linux系统编程(文件、进程线程、进程间通信)

    文章目录 Linux系统编程 文件IO open/close函数 read/write函数 文件描述符 阻塞.非阻塞 fcntl函数 lseek函数 传入传出参数 文件系统 文件存储 文件操作 sta ...

最新文章

  1. fail2ban使用教程
  2. (转载)php array_merge 和 两数组相加区别
  3. 解决fatal: 不是一个 git 仓库(或者任何父目录)的方法
  4. 面向对象发牌Java_面向对象思路的斗地主发牌小程序
  5. C# winform WebBrowser怎么获取js中的变量的值?怎么触发js的事件?
  6. logistic回归 简介_从零实现机器学习算法(四)Logistic回归
  7. 事件冒泡与事件捕获,附实例
  8. Unity3D之Assetbundle
  9. 真倒霉,前不久分區表錯誤把我數據全部搞沒了
  10. Boost Asio Examples(整理)
  11. 算法笔记_面试题_22. 图与搜索(BFS/DFS)相关_模板及示例十几道
  12. java生成 折线图
  13. 图书馆管理系统用例图
  14. 《我的成功可以复制》读后感这一、两天可以静下心来,将唐骏先生写的《我...
  15. typora 免费版, 最后一个beta版本下载
  16. 链接测试工具-Xenu
  17. Markdown详细教程+技巧总结
  18. 基于Java的超市进销存管理系统的设计与实现
  19. 声明式导航与编程式导航
  20. 【国际】塞拉利昂重点发展国家区块链计划

热门文章

  1. 树状数组基本用法详解
  2. matlab中方差直方图,如何规范直方图在MATLAB?
  3. Altiumnbsp;designernbsp;学习教程
  4. excel设置图片自动更新_智能Excel排班表,日期自动更新,三班排班一键统计,极简轻松...
  5. python手机端秒杀_python实现淘宝秒杀脚本
  6. scatter python_python数据可视化(matplotlib、scatter)
  7. 数学--数论--费马小定理+求逆元
  8. USACO Training Section 1.1黑色星期五Friday the Thirteenth
  9. java垃圾回收机制算法分析
  10. Python高级特性: 12步轻松搞定Python装饰器