最全Linux 线程控制_linux 主线程return了 为啥进程还在运行(4),滴滴C C++面试

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

    sleep(1);
}

return 0;

}


当我们运行一个程序时,都会有一个主线程,而这个主线程就是在main函数中的一条执行流,我们可以通过主线程创建普通线程,让它去执行一个函数,完成指定的功能,线程由于都占用同一个虚拟地址空间,共享大部分资源,虚拟地址通过页表映射都是一样的,所以只要可以拿到一个变量的地址,再通过虚拟地址修改物理地址上的内容,当其他线程也使用这个变量时,也是会一样映射到同一块物理内存,这个修改会影响到所有的线程。代码中的两个线程同时执行,当普通线程对全局变量a进行内容修改,主线程使用a是,也是被修改过后的内容了。再看主线程中的内容ptr,通过地址传给普通线程,普通线程也对该变量进行内容修改,主线程使用时,也是使用被修改过的内容  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330104643192.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)  
 什么是tid呢?  
 tid的类型是一个无符号长整形数据,tid是一个线程id,也就是线程的操作句柄,准确来说这个tid是用户态线程的id,在空间中这个id是一块空间的首地址。每个线程被创建出来之后,都会开辟一块空间,存储自己的栈和自己的描述信息  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330125120853.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)


task\_struct-> pid:轻量级进程id,也就是ps -efL看到的LWP  
 task\_struct->tgid:线程组id,等于主线程id,也就是ps -ef看到的进程id)  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330124748269.png)


## 线程终止


#### `return`


线程入口函数运行完毕,线程就会自动退出,在线程入口函数中调用return。(main函数调用return是退出进程)


在主线程中调用return 主线程return退出,全部线程都退出  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330131327617.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)  
 在普通线程中调用return 普通线程退出不影响其他线程的执行  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330131706573.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)


#### `pthread_exit()`


接口`pthread_exit(void *retval)`退出线程的接口,谁调用谁退出,retval是退出返回值


在主线程中调用pthread\_exit 主线程退出,并不会使进程退出,只有所有线程退出,进程才会退出  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330132353653.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)  
 在普通线程中调用pthread\_exit  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330132732828.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)


#### `pthread_cancel()`


接口`int pthread_cancel(pthread_t thread)` 终止一个指定的线程,退出的线程是被动取消的。


![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330133852791.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)


## 线程等待


线程等待就是等待一个进程的退出,获取退出线程的返回值,回收线程所占的资源


每个线程都有一个属性,默认创建出来的这个属性是joinnable,退出后,需要被其他线程等待获取返回值并回收资源


##### `pthread_join()`


接口`int pthread_join(pthread_t thread, void **retval)` 等待指定的线程退出,获取其返回值,这是一个阻塞函数,线程没有退出则一直等待  
 参数内容(`thread`:要等待退出的线程id;`retval`:输出型参数,用于返回线程的返回值)  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330141334295.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)



> 
> 特别要注意的是,返回值必须是字符串指针,不能是字符串数组,字符串数组是在栈区的,当一个线程退出,这个字符串数字也就会被系统收回,此时另一个线程等到返回值时就是收到一块已经释放了的空间的内容
> 
> 
> 


我们知道线程等待是一个会是线程发生阻塞的操作,但是如果我们不进行等待,就会造成资源的泄漏,那我们有没有办法既等待线程,又不阻塞呢?答案是可以的,这时候就要引入**线程分离**


## 线程分离


**线程分离**:将线程joinable属性修改为detach属性  
 一个线程的属性若是joinable那么就必须被等待,若是这个属性是detach,那么这个线程退出后则自动释放资源,不需要被等待(也不能被等待)  
 **使用场景**:对获取一个线程退出的返回值没有意义,且不想等待线程退出  
 接口`int pthread_detach(pthread_t thread)`,将指定的线程分离出去----将属性修改为detach  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330143614656.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)  
 线程分离并非需要其他线程来设定,每个线程都可以将自己进行线程分离,也就是在函数中调用pthread\_detach()函数,但是我们如何获取自己线程的tid呢?库函数中有一个接口`pthread_t pthread_selt(void)` 返回调用线程的pid  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210330144225764.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQ0NDQzOTg2,size_16,color_FFFFFF,t_70)


![img](https://img-blog.csdnimg.cn/img_convert/90ff4107082103f867db34ba6fab8c2c.png)
![img](https://img-blog.csdnimg.cn/img_convert/bcaa5fc673b6f11ab0c4de9d34560fca.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618668825)**

出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618668825)**

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值