【Linux】线程——线程的概念、线程的特点、线程的优点和缺点、线程和进程、线程函数的使用

Linux线程

1. 线程的概念

  我们之前认识到,进程是程序的一个执行实例,是正在执行的程序等,这是从进程本身来看的,从内核上看,进程担当分配系统资源(CPU时间,内存)的实体,是联系硬件和软件的桥梁。

  通常进程包含:进程独立的程序地址空间和页表,通常是我们说的虚拟内存;
进程拥有系统分配的各种资源;标识进程唯一性和保存执行信息的进程控制块(PCB);执行的程序代码和相关的数据;打开的文件和设备;进程上下文。

  当一个进程要执行大量任务的时候,一个单进程通常只有一个执行流,只能执行一个时间片,这样无法提高计算机的运行效率,所以我们要在进程中使用线程。

  通过线程,我们可以提高计算机的运行效率。线程是进程中的执行单元,多个线程可以共享进程的资源,如内存空间、文件描述符等。这样使得多个线程可以在同一个进程中并发执行,将复杂的任务分解为多个子任务,每一个子任务有线程执行,充分利用CPU的时间片,减少等待时间,提高系统运行效率。

  简单总结:

  进程因为包含很多资源,开销较大,线程在进程内部开销较小。

  多线程有多个时间片且并发执行,资源共享,运行效率较高。

  

1.1 什么是线程

  (1)在一个程序里的一个执行路线就叫做线程(thread)。更准确的定义是:线程是“一个进程内部的控制序列”。

  (2)一切进程至少都有一个执行线程。

  (3)线程在进程内部运行,本质是在进程地址空间内运行。

  (4)在Linux系统中,在CPU眼中,看到的PCB都要比传统的进程更加轻量化,所以Linux 下的线程看作轻量级进程。

  (5)透过进程虚拟地址空间,可以看到进程的大部分资源,将进程资源合理分配给每个执行流,就形成了线程执行流。

  总结:线程是进程中的一个实体,作为系统调度和分派的基本单位。

在这里插入图片描述

  

2. 线程的特点

2.1 线程的优点

  (1)创建一个新线程的代价要比创建一个新进程小得多

  (2)与进程之间的切换相比,线程之间的切换需要操作系统做的工作要少很多

  (3)线程占用的资源要比进程少很多

  (4)能充分利用多处理器的可并行数量

  (5)在等待慢速I/O操作结束的同时,程序可执行其他的计算任务

  (6)计算密集型应用,为了能在多处理器系统上运行,将计算分解到多个线程中实现

  (7)I/O密集型应用,为了提高性能,将I/O操作重叠。线程可以同时等待不同的I/O操作。

  

2.2 线程的缺点

  (1)性能损失:一个很少被外部事件阻塞的计算密集型线程往往无法与共它线程共享同一个处理器。如果计算密集型线程的数量比可用的处理器多,那么可能会有较大的性能损失,这里的性能损失指的是增加了额外的同步和调度开销,而可用的资源不变。

  (2)健壮性降低:编写多线程需要更全面更深入的考虑,在一个多线程程序里,因时间分配上的细微偏差或者因共享了不该共享的变量而造成不良影响的可能性是很大的,换句话说线程之间是缺乏保护的。

  (3)缺乏访问控制:进程是访问控制的基本粒度,在一个线程中调用某些OS函数会对整个进程造成影响。

  (4)编程难度提高:编写与调试一个多线程程序比单线程程序困难得多

  

2.4 线程和进程

进程和线程的区别:

  (1)调度单位:进程是资源分配的基本单位,线程是调度的基本单位。

  (2)资源拥有:进程拥有独立的地址空间和系统资源,线程共享进程资源,但也拥有自己的一部分资源:线程ID、一组寄存器、栈、errno、信号屏蔽字、调度优先级。

  (3)系统开销:创建或撤销进程时,系统需要分配或回收资源,开销较大;线程的创建和撤销开销相对较小。

  (4)通信方式:进程间通信较为复杂,需要使用特定的进程间通信机制,如管道、消息队列等;线程间通信相对简单,可以通过共享内存等方式直接进行。

  (5)健壮性:进程间相互独立,一个进程的崩溃一般不会影响其他进程;线程一个线程的崩溃可能影响整个进程。

  此外因为进程的多个线程共享同一地址空间,因此Text Segment、Data Segment都是共享的,如果定义一个函数,在各线程中都可以调用,如果定义一个全局变量,在各线程中都可以访问到,除此之外,各线程还共享以下进程资源和环境:文件描述符表,每种信号的处理方式(SIG_ IGN、SIG_ DFL或者自定义的信号处理函数)当前工作目录,用户id和组id。

在这里插入图片描述

  

3. 线程函数的使用

  POSIX线程库:与线程有关的函数构成了一个完整的系列,绝大多数函数的名字都是以“pthread_”打头的,要使用这些函数库,要通过引入头文<pthread.h>,链接这些线程函数库时要使用编译器命令的“-lpthread”选项。

pthread_create() 创建线程

功能:创建一个新的线程

原型:

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

参数:

  thread:返回线程ID
  attr:设置线程的属性,attr为NULL表示使用默认属性
  start_routine:是个函数地址,线程启动后要执行的函数
  arg:传给线程启动函数的参数

返回值:

  成功返回0;失败返回错误码

错误检查:

  传统的一些函数是,成功返回0,失败返回-1,并且对全局变量errno赋值以指示错误。

  pthreads函数出错时不会设置全局变量errno(而大部分其他POSIX函数会这样做)。而是将错误代码通过返回值返回pthreads同样也提供了线程内的errno变量,以支持其它使用errno的代码。

  对于pthreads函数的错误,建议通过返回值判定,因为读取返回值要比读取线程内的errno变量的开销更小。

  

进程地址空间布局

  pthread_ create函数会产生一个线程ID,存放在第一个参数指向的地址中。该线程ID和前面说的线程ID不是一回事。

  前面讲的线程ID属于进程调度的范畴。因为线程是轻量级进程,是操作系统调度器的最小单位,所以需要一个数值来唯一表示该线程。

  pthread_ create函数第一个参数指向一个虚拟内存单元,该内存单元的地址即为新创建线程的线程ID,属于NPTL线程库的范畴。线程库的后续操作,就是根据该线程ID来操作线程的。

  线程库NPTL提供了pthread_ self函数,可以获得线程自身的ID。

  pthread_t 到底是什么类型呢?取决于实现。对于Linux目前实现的NPTL实现而言,pthread_t类型的线程ID,本质就是一个进程地址空间上的一个地址。

在这里插入图片描述

  

pthread_self() 获取线程ID

pthread_t pthread_self(void
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

鳄鱼麻薯球

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值