参照:线程与进程,你真得理解了吗_云深i不知处的博客-CSDN博客
1 进程
- 进程是系统分配资源的最小单位
- 进程拥有>= 1个线程
- 每个进程用用独立的地址空间;包括:代码区,数据区,堆栈区,进程之间的地址空间是隔离的,互不影响。
2 进程的切换
进程的切换会带来额外的开销:即上下文(执行B线程,需要保留A线程的线程,执行的时候在拿出来)切换。
- 上下文保存具体的是:1.CPU当前所有寄存器中的值 2.保存进程所有状态:当前进程的状态 3.保存进程的堆栈。
3 进程的运行
进程的切换涉及并发与并行
- 并发:每个进程被操作系统分配一个时间片:每次CPU分配给这个进程可用的时间长度。这么做是因为只有一个单核的CPU,但需要执行多个进程。因为CPU的切换和进程的执行非常快,所以看起来是多个进程“并行运行”
- 并行:随着多核CPU的出现,并行让真正的多个任务同时执行得以实现。
4 线程
- 同一个进程中的线程共享进程的资源
- 线程有自己的栈区,共享进程的数据区,代码区,堆区。
5 函数在内存中的样子(摘自:函数运行时在内存中是什么样子? - 知乎)
- 函数的调用是在栈中进行的。
- 函数调用过程中函数栈详解_芹泽的博客-CSDN博客_函数栈
5.1.1函数的调用与返回
- 通过函数A中对应的函数B地址,从此处跳转进B执行。对应代码中所写的函数调用,即 call 0x400540 (注:0x400540就是函数B的地址)
- 将函数A中执行完函数B()之后的下一条指令地址push到A的栈帧,即 0x40056a
- .函数B执行完代码,然后再执行最后一条指令:ret , 这样函数B执行完了就跳转到函数A继续执行。跳转到的地址就是第2步的0x40056a
5.1.2 函数B的参数传入
- 当从函数A()跳转到函数B()的时候,如果需要传递一些参数,这是如何实现的呢?函数的传入参数本是存在寄存器的,但当传入的参数过多,参数就会被写入栈帧。
- 函数B传入的参数是函数A的局部变量,所以参数是存在函数A的栈帧里面的。
- 这里也有个面试点:函数参数的入栈顺序:函数的参数是 由右向左入栈的
6 进程运行时的地址空间(摘自:线程是如何共享进程资源的? - 知乎)
- 栈区
线程之间不乱整,肯定是私有的,但如果A线程的局部变量a, B线程可以拿到a,B也可以修改a
- 堆区:代码通过malloc/new出来的内存放的位置
只要知道堆区的指针,每个线程都可访问指针指向的数据。所以线程时共享堆区的
- 代码区:写的代码编译成的可执行的机器指令。
线程共享代码区,每个线程都可使用代码区的所有函数
- 数据区
同理:线程之间共享。
综上:线程共享进程的资源
6 线程共享了哪些资源
- 同一个进程里的线程共享这个所属进程的地址空间所有的内容。
- 线程运行的本质就是函数的执行,而且每个线程有自己独立,私有的栈区。同时函数运行需要额外的寄存器来保存像寄存器之类的信息。
- 线程私有:栈区,程序计数器,栈指针,运行所需的寄存器。这些可统称为线程上下文