什么是线程?
linux内核中是没有线程这个概念的,而是轻量级进程的概念:LWP。一般我们所说的线程概念是C库当中的概念。
1.1线程是怎样描述的?
线程实际上也是一个task_struct,工作线程拷贝主线程的task_struct,然后共用主线程的mm_struct。线程ID是在用task_struct中pid描述的,而task_struct中tgid是线程组ID,表示线程属于该线程组,对于主线程而言,其pid和tgid是相同的,我们一般看到的进程ID就是tgid。
即:
获取线程ID和主线程ID的值:
但是获取该gettid系统调用接口并没有被封装起来,如果确实需要获取线程ID,可使用:
#include <sys/syscall.h>
int TID = syscall(SYS_gettid);
则对线程组而言,所有的tgid一定是一样的,所有的pid一定是不一样的。主线程pid和tgid一样,工作线程pid和tgid一定不一样。
1.2如何查看一个线程的ID
命令:ps -eLf
上述polkitd进程是多线程的,进程ID为731,进程内有6个线程,线程ID为731,764,765,768,781,791。
1.3多线程如何避免调用栈混乱的问题?
工作线程和主线程共用一个mm_struct,如果都向栈中压栈,必然会导致调用栈出错。
实际上工作线程压栈是压了共享区,该共享区包含了许多线程独有的资源。如图:
每一个线程,默认在共享区中占有的空间为8M,可以使用ulimit -s修改。
进程是资源分配的基本单位,线程是调度的基本单位。
1.3.1线程独有资源
- 线程ID
- 一组寄存器
- errno
- 信号屏蔽字
- 调度优先级
1.3.2线程共享资源和环境
- 文件描述符表
- 信号的处理方式
- 当前工作目录
- 用户id和组id