Linux--线程安全、多线程fork问题

目录

一、概念:

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

1.代码:

2.结果:

3.解决方法:

三、多线程fork()

1.代码:

2.线程id

 3.增加fork()代码:

4.改变fork()位置

5.运行结果

6.总结

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

1.代码:

//运行结果

子进程为什么加不上锁?

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


一、概念:

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

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

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

1.代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>

void* 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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>

void* 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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>

void* 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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>

void* 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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>

void* 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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>
#include<sys/wait.h>

pthread_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<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>
#include<sys/wait.h>

pthread_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程序

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/2147.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【嵌入式硬件芯片开发笔记】HART协议调制解调芯片AD5700配置流程

【嵌入式硬件芯片开发笔记】HART协议调制解调芯片AD5700配置流程 XTAL_EN接地&#xff0c;CLK_CFG的两个引脚由同一个GPIO控制 初始时HART_CLK_CFG输出低电平 由RTS引脚控制调制/解调。当RTS处于高电平时&#xff0c;为解调&#xff08;输入&#xff09;&#xff1b;否则为调…

【计算机组成原理】:计算机系统概述

目录 一、计算机系统层次结构 1️⃣计算机系统的组成 2️⃣计算机硬件 1. 冯诺依曼机的基本思想 &#x1f4a4;思考&#xff1a;冯诺依曼机的来源❓ &#x1f338;知识点&#xff1a;冯诺依曼机的特点 &#x1f4a4;思考&#xff1a;以运算器为中心的计算机有什么缺点❓ 2…

【微服务】—— 统一网关Gateway

文章目录1. 概述1.1 为什么需要网关1.2 SpringCloud Gateway2. gateway快速入门搭建网关服务1、创建新的module&#xff0c;引入SpringCloudGateway的依赖和nacos的服务发现依赖&#xff1a;2、编写路由配置和nacos地址3. 断言工厂路由断言工厂Route Predicate Factory4. 过滤器…

【数据结构】千字深入浅出讲解队列(附原码 | 超详解)

&#x1f680;write in front&#x1f680; &#x1f4dd;个人主页&#xff1a;认真写博客的夏目浅石. &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd; &#x1f4e3;系列专栏&#xff1a;C语言实现数据结构 &#x1f4ac;总结&#xff1a;希望你看完…

javaweb实习实训管理系统mysql

本毕业设计基于JSP的实习实训管理系统&#xff0c;本系统能实现网上的实习实训信息管理&#xff0c;主要功能有&#xff1a;添加用户、查看用户、管理用户、添加实验室、查看实验室、管理实验室、添加课程、查看课程、管理课程、添加教学、查看教学、管理教学、添加实习、查看实…

STM32的推挽输出和开漏输出

文章目录 前言一、推挽输出二、开漏输出三、区别和适应场景总结前言 本篇文章将带大家了解STM32的推挽输出和开漏输出,并且学习这两个的区别,学习分别在什么时候使用这两个不同的输出方式。 在 STM32 微控制器中,GPIO(General Purpose Input/Output)模块是一个通用的输入…

Java 到底是值传递还是引用传递?

C 语言是很多变成语言的母胎&#xff0c;包括 Java。对于 C 语言来说&#xff0c;所有的方法参数都是通过 “值” 传递的&#xff0c;也就是说&#xff0c;传递给被调用方法的参数值存放在临时变量中&#xff0c;而不是存放在原来的变量中。这就意味着&#xff0c;被调用的方法…

项目质量管理工作 不得不重视的4大关键点

1、三大视角确保项目质量 我们需要从客户视角、SOW视角和组织视角三大视角&#xff0c;确保项目的质量。 从客户视角方面出发&#xff0c;满足客户的要求&#xff0c;如项目交付的准时性、项目质量的保证等。我们需要全力保障客户对项目质量的要求。 从SOW视角确保项目质量&…

[ 漏洞复现篇 ] Joomla未授权访问Rest API漏洞(CVE-2023-23752)

&#x1f36c; 博主介绍 &#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 _PowerShell &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【数据通信】 【通讯安全】 【web安全】【面试分析】 &#x1f389;点赞➕评论➕收藏 养成习…

Java实习生------MySQL10道面试题打卡

今日语录&#xff1a;“没有执行力&#xff0c;就没有竞争力 ”&#x1f339; 参考资料&#xff1a;图解MySQL、MySQL面试题 1、事务有哪些特性&#xff1f; 原子性&#xff1a; 一个事务中的所有操作&#xff0c;要么全部完成&#xff0c;要么全部不完成&#xff0c;不会出现…

Three.js——learn01

Three.js——learn01Three.js——learn01本地搭建文档通过parcel搭建Threejs环境1.初始化2.安装parcel设置打包位置3.设置目录结构QuickStart安装threejsindex.htmlindex.cssindex.js启动Three.js——learn01 本地搭建文档 登录GitHub搜索three.js git clone https://github…

基于数据安全的沙盘推演体系

背景 2022年由IBM和Ponemon研究所联合发布的一份全球性的研究报告&#xff0c;分析了550家遭受数据泄露事件的组织的各种成本和影响因素。根据报告&#xff0c;2022年全球数据泄露规模和平均成本均创下历史新高&#xff0c;数据泄露事件的平均成本高达435万美元&#xff0c;比2…

C语言—文件操作

1.为什么使用文件使用文件可以直接将数据存放到电脑硬盘上&#xff0c;做到数据的持久化2.什么是文件硬盘上的文件是文件但在程序中&#xff0c;我们一般谈的文件有两种&#xff1a;程序文件和数据文件&#xff08;从文件功能角度来分类的&#xff09;2.1程序文件包括源程序文件…

vue3使用vee-validate自定义表单校验,提交实现步骤

1、首先安装vee-validate&#xff08;指定版本&#xff09;&#xff0c;安装命令如下&#xff1a; npm i vee-validate4.0.32、在app.vue中写入如下内容&#xff1a;用vee-validate提供的Form组件代替form标签&#xff0c;用Field组件代替input标签&#xff0c;errors是接收校…

UnixBench----x86架构openEuler操作系统上进行性能测试

【原文链接】UnixBench----x86架构openEuler操作系统上进行性能测试 &#xff08;1&#xff09;打开github上 UnixBench 地址&#xff0c;找到发布的tag &#xff08;2&#xff09;找到tar.gz包&#xff0c;右键复制链接 比如这里是 https://github.com/kdlucas/byte-unix…

1、OSI模型

目录 一、OSI模型 二、TCP / IP 模型 (协议簇&#xff09; 1、TCP/IP简介 2、自下而上了解TCP/IP协议&#xff1a; &#xff08;网络接口和物理层&#xff09; 3、TCP/IP协议其他知识点 三、基本知识点 1、socket——插座 2、为什么需要socket 3、什么是socket 4、IP地…

【数据结构】夯实基础|线性表刷题01

作者&#xff1a;努力学习的大一在校计算机专业学生&#xff0c;热爱学习和创作。目前在学习和分享&#xff1a;算法、数据结构、Java等相关知识。博主主页&#xff1a; 是瑶瑶子啦所属专栏: 【数据结构|刷题专栏】&#xff1a;该专栏专注于数据结构知识&#xff0c;持续更新&a…

【三维几何学习】从零开始网格上的深度学习-3:Transformer篇(Pytorch)

本文参加新星计划人工智能(Pytorch)赛道&#xff1a;https://bbs.csdn.net/topics/613989052 从零开始网格上的深度学习-3:Transformer篇引言一、概述二、核心代码2.1 位置编码2.2 网络框架三、基于Transformer的网格分类3.1 分类结果3.2 全部代码引言 本文主要内容如下&#…

linux中写定时任务

场景&#xff1a;我们生产环境中有大量的日志记录&#xff0c;但是我们的磁盘没有太大&#xff0c;需要定时清理磁盘 文章目录crond 定时任务详解安装定时任务crontab服务启动与关闭crontab操作crontab 命令test.sh查看日志丢弃linux中的执行日志Linux进入nano模式方式一方式二…

Unreal Engine 网络系统(四):UEC++的RPC

目录 行为同步 On Server&#xff1a;服务端的RPC代码 On Client&#xff1a;客户端的RPC代码 NetMulticast&#xff1a;广播的RPC代码 属性同步 行为同步 借助UFUNCTION进行函数标记 UFUNCTION(Server)&#xff1a;声明一个在客户端调用&#xff0c;在服务端执行的函数U…