Linux--线程安全、多线程fork问题
创始人
2025-05-30 16:47:39

目录

一、概念:

二、利用空格分割字符串:

1.代码:

2.结果:

3.解决方法:

三、多线程fork()

1.代码:

2.线程id

 3.增加fork()代码:

4.改变fork()位置

5.运行结果

6.总结

 四、父进程有锁,fork()后子进程有锁吗

1.代码:

//运行结果

子进程为什么加不上锁?

//如何解决子进程被锁住?


一、概念:

线程安全:多线程无论调度顺序怎么样,都可以得到正确的结果(由并发线程引起)

同步:线程安全的函数,可重入函数

二、利用空格分割字符串:

1.代码:

#include
#include
#include
#include
#includevoid* fun(void* arg)
{char buff[]={"a b c d e f g h"};char* s=strtok(buff," ");while(s!=NULL){   printf("fun s=%s\n",s);sleep(1);s=strtok(NULL," ");}   
}int main()
{pthread_t id; pthread_create(&id,NULL,fun,NULL);char arr[]="1 2 3 4 5 6 7 8";char* s=strtok(arr," ");while(s!=NULL){   printf("main s=%s\n",s);sleep(1);s=strtok(NULL," ");}   pthread_join(id,NULL);exit(0);
}

2.结果:

 //为什么会出现这种状况?

一个指针,只能记住一个位置,main在打印1之后睡眠一秒钟,回去找指针的时候,指针已经指向b了,所以fun和main同时打印buff,两个线程同时获取一个字母的时候,快的将空格改成\0,让线程误以为结束了,把指针赋空了,进程就结束了。

3.解决方法:

使用strtok_r(_r代表线程安全版本)

//代码

#include
#include
#include
#include
#includevoid* fun(void* arg)
{char buff[]={"a b c d e f g h"};char* ptr=NULL;char* s=strtok_r(buff," ",&ptr);while(s!=NULL){   printf("fun s=%s\n",s);sleep(1);s=strtok_r(NULL," ",&ptr);//NULL的作用只是为了使得每次调用时,都不是从头开始,而是从上次调用时查找所停止的位置开始,}   
}int main()
{pthread_t id; pthread_create(&id,NULL,fun,NULL);char arr[]="1 2 3 4 5 6 7 8";char* ptr=NULL;char* s=strtok_r(arr," ",&ptr);while(s!=NULL){   printf("main s=%s\n",s);sleep(1);s=strtok_r(NULL," ",&ptr);}   pthread_join(id,NULL);exit(0);
}

//运行结果

 //安全

为什么NULL可以达到这个作用?
答:strtok里面有静态变量指针,会记住分割到哪里,传空就是沿着当前位置继续分割,如果不是空,那就分割你传进来的字符串

三、多线程fork()

1.代码:

#include
#include
#include
#include
#includevoid* fun(void*arg)
{for(int i=0;i<5;i++){printf("fun run pid=%d\n",getpid());sleep(1);}
}int main()
{pthread_t id;pthread_create(&id,NULL,fun,NULL);for(int i=0;i<5;i++){printf("main run pid=%d\n",getpid());sleep(1);}
}

2.线程id

 ps -eLf

 //主线程id作为整个进程的id, 2代表的有两个线程

 3.增加fork()代码:

#include
#include
#include
#include
#includevoid* fun(void*arg)
{for(int i=0;i<5;i++){printf("fun run pid=%d\n",getpid());sleep(1);}
}int main()
{pthread_t id;pthread_create(&id,NULL,fun,NULL);fork();for(int i=0;i<5;i++){printf("main run pid=%d\n",getpid());sleep(1);}
}

 fork();//后面的代码在子进程执行还是父进程执行?

都执行

//运行结果

//pid=11622去哪了?

被线程消耗了 ,打印的进程id

4.改变fork()位置

#include
#include
#include
#include
#includevoid* fun(void*arg)
{fork();for(int i=0;i<5;i++){   printf("fun run pid=%d\n",getpid());sleep(1);}   
}int main()
{pthread_t id; pthread_create(&id,NULL,fun,NULL);for(int i=0;i<5;i++){   printf("main run pid=%d\n",getpid());sleep(1);}   
}

5.运行结果

6.总结

当一个进程有多条执行路径,对他进行fork(),产生的子进程只有一条执行路径,被所在线程启用

 四、父进程有锁,fork()后子进程有锁吗

1.代码:

#include
#include
#include
#include
#include
#includepthread_mutex_t mutex;void* fun(void* arg)
{pthread_mutex_lock(&mutex);printf("fun lock\n");sleep(5);pthread_mutex_unlock(&mutex);printf("fun unlock\n");
}int main()
{pthread_t id;pthread_mutex_init(&mutex,NULL);pthread_create(&id,NULL,fun,NULL);sleep(1);pid_t pid=fork();if(pid==-1){exit(0);}if(pid==0)//在子进程中{printf("子进程将要加锁\n");pthread_mutex_lock(&mutex);printf("子进程加锁成功\n");pthread_mutex_unlock(&mutex);}else//在父进程中{wait(NULL);//等待子进程结束,如果子进程没有结束,那么会阻塞printf("main over\n");}exit(0);
}

//运行结果

//子进程加锁没成功,执行到加锁的地方,父进程加锁解锁完成执行在wait

子进程为什么加不上锁?

//二次加锁,证明父进程加锁时,fork()也有锁,即就是fork()后有两个锁

如果是一个锁,可以加锁成功

fork()时,父进程的锁是什么状态,子进程复制出来的锁就是什么状态

//如何解决子进程被锁住?

不可以把锁删掉(存在临界资源等问题)

选择没人用锁的时候进行(直接加锁,验证锁处于什么状态,加锁成功,再解锁)

但是如果别人正在用,那么会堵塞一段时间(锁住的时间)

//代码

#include
#include
#include
#include
#include
#includepthread_mutex_t mutex;void* fun(void* arg)
{pthread_mutex_lock(&mutex);printf("fun lock\n");sleep(5);pthread_mutex_unlock(&mutex);printf("fun unlock\n");
}void at_lock(void)
{pthread_mutex_lock(&mutex);
}
void at_unlock(void)
{pthread_mutex_unlock(&mutex);
}int main()
{pthread_t id;pthread_atfork(at_lock,at_unlock,at_unlock);pthread_mutex_init(&mutex,NULL);pthread_create(&id,NULL,fun,NULL);sleep(1);pid_t pid=fork();if(pid==-1){exit(0);}if(pid==0)//在子进程中{printf("子进程将要加锁\n");pthread_mutex_lock(&mutex);printf("子进程加锁成功\n");pthread_mutex_unlock(&mutex);}else//在父进程中{wait(NULL);//等待子进程结束,如果子进程没有结束,那么会阻塞printf("main over\n");}exit(0);
}

 //结果

//父进程解锁之后才复制,所以子进程复制的锁也是处于解开的状态,所以子进程可以加锁再解锁成功,最后得到退出码,结束main程序

相关内容

热门资讯

数据结构---队列 专栏:数据结构 个人主页:HaiFan. 专栏简介:这里是...
数字操作方法 系列文章目录 前端系列文章——传送门 JavaScript系列文章——传送门 文章目录系列文章目录...
Cartesi 2023 年 ... 查看 Cartesi Machine、Cartesi Rollups 和 Noether 的更新正在...
JavaWeb——jsp概述入... JSP定义:  在如下一个jsp文件里面有如下的代码  <%@ page content...
一切喜怒哀乐都来自于你的认知 01 有个学子,准备出国,父母请来清华的教授宁向东。请问教授࿱...
JAVA并发编程——synch... 引言         Java语言为了解决并发编程中存在的原子性、可见性和有序性问题,...
git学习----3.21 未... 文章目录前言Git :一个分布式版本控制工具目标一、概述1.1 开发中的实际场景1.2...
Qt优秀开源项目之十七:QtP... QtPromise是Promises/A+规范的Qt/C++实现。该规范的译...
【前端八股文】JavaScri... 文章目录Set概念与arr的比较属性和方法并集、交集、差集Map概念属性和方法String用索引值和...
海康硬盘录像机接入RTSP/o... EasyNVR安防视频云服务平台可支持设备通过RTSP/Onvif协议接入平台,能提供...
在混合劳动力时代如何避免网络安... 在混合劳动力时代如何避免安全网络风险 三年多来,混合工作一直是工作生活中不可或缺的一...
2023还不懂Jmeter接口... 这里介绍的Jmeter接口测试的的实战,如果文章内容没遇看懂的话,我这边...
基于4G/5G弱网聚合的多链路... 基于4G/5G多卡聚合(弱网聚合)的智能融合通信设备技术亮点 增强带宽提供可靠连接 通过将多个有线和...
如何使用Synplify综合v... 文章目录使用Synplify综合的好处synplify的教程方法1(无效)...
2023年全国最新高校辅导员精... 百分百题库提供高校辅导员考试试题、辅导员考试预测题、高校辅导员考试真题、辅导员证考试题库等ÿ...
2022年18个值得期待的Le... 有数百个独特的LearnDash附加组件,您可能很难选择您的LearnDash LMS...
【java基础】Stream流... 文章目录基本介绍流的创建流的各种常见操作forEach方法filter方法map方法peek方法fl...
javaweb高校行政办公自动... 本课题基于我国高校管理信息化建设现状,结合在实际工作中所遇到的问题和收获,...
一款专门为自动化测试打造的集成... 你好,我是不二。 随着行业内卷越来越严重,自动化测试已成为测试工程师的...
【go-zero】golang... 一、casbin 概览 1、casbin基本了解 casbin的GitHub:https://git...
现在开发低代码平台算晚吗? 现在开发低代码平台算晚吗?作为低代码的亲戚——零代码厂商,这篇就以“厂商...
【JavaWeb】书城项目(2... 222.书城项目-第三阶段:修改所有html页面为jsp页面 改成jsp页面之后&#x...
基于jeecgboot的大屏设...      通过前面设计好数据源后,就要进行数据集的设计了。      一、还是在onl...
Linux命令小技巧:显示文件... 工作中会有很多千奇百怪的需求,比如:如何在 Linux 命令行中快速找到...
【找工作】-- 大数据工程师找... 目录 1.前言 2.找工作的理论知识 2.1 分析个人特征 2.1.1 你自身优势是什么?
C++基础算法④——排序算法(... 排序算法 1.插入排序 2.桶排序 1.插入排序 基本思想:将初始数据分为有序部分和...
nginx快速入门.跟学B站n... nginx快速入门.跟学B站nginx一小时精讲课程笔记nginx简介及环境准备nginx简介环境准...
ORACLE存过互相调用之间事... 今天在问答区看到一个问题是 假如有procedureA、procedureB和procedureC&...
基于java中Springbo... 基于java中Springboot框影视影院订票选座管理系统 开发语言:Java 框...
CVE-2018-18086 最近闲来无事,看到青少年CTF平台,感觉对新手还是比较友好的࿰...