Linux学习记录——이십삼 进程信号(2)

news2025/1/10 21:08:35

文章目录

  • 1、可重入函数
  • 2、volatile关键字
  • 3、如何理解编译器的优化
  • 4、SIGCHLD信号


1、可重入函数

两个执行流都执行一个函数时,这个函数就被重入了。比如同一个函数insert,在main中执行时,这个进程时间片到了,嵌入了内核,然后再执行另一个函数,这个函数里也调用了insert,当它执行完代码后,又回到main,继续刚才的位置,这就相当于insert被重复进入了,会导致一些问题。这也就说明insert是不可重入函数,如果重入后没有问题,那就是可重入函数。可重入是函数的一个特性。

如果一个函数内部使用了全局性的数据结构,那么这个函数通常是不可重入的;比如malloc,标准IO库的大多数函数都是不可重入含糊。如果一个函数里只有一些局部变量,那就可以重入

2、volatile关键字

#include <stdio.h>
#include <signal.h>

int quit = 0;

void handler(int signo)
{
    printf("change quit from 0 to 1\n");
    quit = 1;
}

int main()
{
    signal(2, handler);
    while(!quit);//这里故意没有携带while的代码块,故意让编译器认为在main中,quit只会被检测
    printf("man quit 正常\n");
    return 0;
}

像这样的代码,可以发现如果如果不执行handler,while那里就会死循环,而执行handler后,while就结束,就打印正常的语句。

但是实际上gcc编译的时候是有一些编译级别的,它就做优化。

在这里插入图片描述

mysignal:mysignal.c
    gcc -o mysignal mysignal.cc -O2
.PHONY:clean
clean:
    rm -f mysignal

每个编译器做的优化不一样,分别试试O1 O2 O3 O0等这些选项,运行起来后当我们发送2号信号,也就是Ctrl + C,如果能退出,打印man quit正常,那就说明你的编译器优化级别是当前设置的这个级别。

这怎么优化得?这样一份简单的代码为什么要做优化呀?如果按Ctrl + C没有退出的时候,一直按,一直会打印change quit from 0 to 1,但quit貌似没有变成1,要不然就退出了。所以现在有很多问题,慢慢来。

CPU的运算有两种,算术运算和逻辑运算,这两个顾名思义。所有的运算都要在CPU里,那么到了while循环这里,存在物理内存的quit全局变量要load到CPU的寄存器,交给CPU去做逻辑判断,把结果告知while,CPU里还有PC指针指向这个程序的代码和数据,如果while条件成立,那就继续执行while的代码,如果不成立,PC指针就指向下一行代码,也就是printf那里。

改了信号处理动作后,收到信号,quit就变成1了,然后再次load到内存,再判断,就会退出了。

对于这个代码,编译器对它的优化在于,main函数块里只是判断quit,每一次while判断,判断完后又回到代码,然后再次将quit放到寄存器中,再去判断,所以编译器就在编译代码时把quit放进寄存器后就只在寄存器判断,不再回到内存中再次load了。这也就是编译器对它的优化,以后只需要看寄存器中的数据就好了。所以quit即使在内存中被改变了,寄存器也看不到。

所以编译器优化不一定好,CPU只看到寄存器的数据而看不到内存的,这就是内存位置不可见。那么为了让CPU每一次都从内存中读取数据,不用寄存器的数据,就得用volatile关键字。

volatile int quit = 0;

这个关键字的作用就是保证内存可见性。

3、如何理解编译器的优化

编译器的本质是在代码上动手脚。

CPU其实很笨,用户给什么就执行什么代码。相反操作系统才是很聪明的。

刚才的代码中,如果没优化,代码在转为汇编代码前,先把内存的某个数据先放到寄存器中,再放到另一个寄存器中,然后逻辑反一下,再检测后面放的那个寄存器中的数据即可,条件满足就跳转到循环,继续检测,如果不满足就运行后面的代码;如果优化了,那么循环就从逻辑反开始,而不是load内存数据开始,所以重复循环,它只能看到最一开始的那个数据,即使后面数据改了,也看不到了,因为会发现在循环中不需要改数据,改数据是在内存中改的,所以就直接这样优化了。

4、SIGCHLD信号

在之前,我们知道父进程会阻塞式等待或者非阻塞式轮询来得知子进程退出,这两个方法都需要父进程主动检测,这是因为子进程退出了,父进程不知道。但是这可不是因为子进程只管自己,什么也不说地退出了。

子进程会在结束进程后给父进程发送SIGCHLD信号,只不过父进程默认忽略这个信号。

waitpid的第一个参数也可以这样写。

在这里插入图片描述

测试代码:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <uhistd.h>
#include <signal.h>
pid_t id;

void handler(int signo)
{
    sleep(5);
    printf("捕捉到一个信号: %d, who: %d\n", signo. getpid());
    pid_t res = waitpid(-1, NULL, 0);
    if(res > 0)
    {
        printf("wait success, res: %d, id: %d\n", res, id);
    }
}

int main()
{
    signal(SIGCHLD, handler);
    id = fork();
    if(id == 0)
    {
        int cnt = 5;
        while(cnt)
        {
            printf("我是子进程,我的pid: %d, ppid: %d\n", getpid(), getppid());
            sleep(1);
            --cnt;
        }
        exit(1);
    }
    while(1)
    {
        sleep(1);
    }
    return 0;
}

另开一个窗口,用这行命令while :; do ps axj | head -1 && ps axj | grep mysignal; echo “################”; sleep 1; done,每隔一行检查一下,ehco后双引号的内容可自定义。

但是这个代码有问题,如果说子进程是循环创建了10个,那么这10个子进程依次退出,退出的时候第一个把父进程的pending位对应位置变为1,但是父进程忽略,然后第二个也重复置为1,这就相当于第一个的信号丢失了,这个代码运行起来后,就会发现只有几个被回收了。所以handler里面的回收也要有循环。

    while(1)
    {
        pid_t res = waitpid(-1, NULL, 0);
        if(res > 0)
        {
            printf("wait success, res: %d, id: %d\n", res, id);
        }
        else break;
    }

    int i = 1;
    for(; i <= 10; ++i)
    {
        id = fork();
        if(id == 0)
        {
            int cnt = 5;
            while(cnt)
            {
                printf("我是子进程,我的pid: %d, ppid: %d\n", getpid(), getppid());
                sleep(1);
                --cnt;
            }
            exit(1);
        }
    }

-1的意思就是等待任意一个子进程,直到没有可回收的就结束了。但如果不是所有的进程都会退呢?比如每个进程的运行时间不一样,有5个退了,5个没退,回收5个后,第6个还会继续回收吗?还会继续回收,这样就变回了阻塞式等待。把waitpid的0换成WNOHANG,也就是非阻塞式。

要想不产生僵尸进程还有另外一种方法,父进程调用sigaction将SIGCHLD的处理动作置为SIG_IGN,这样fork出来的子进程在终止时会自动清理掉。

整体代码:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <uhistd.h>
#include <signal.h>
pid_t id;

void handler(int signo)
{
    printf("捕捉到一个信号: %d, who: %d\n", signo. getpid());
    sleep(5);
    while(1)
    {
        pid_t res = waitpid(-1, NULL, WNOHANG);
        if(res > 0)
        {
            printf("wait success, res: %d, id: %d\n", res, id);
        }
        else break;
    }
    printf("handler done\n");
}

int main()
{
    //signal(SIGCHLD, handler);
    signal(SIGCHLD, SIG_IGN);
    int i = 1;
    for(; i <= 10; ++i)
    {
        id = fork();
        if(id == 0)
        {
            int cnt = 5;
            while(cnt)
            {
                printf("我是子进程,我的pid: %d, ppid: %d\n", getpid(), getppid());
                sleep(1);
                --cnt;
            }
            exit(1);
        }
    }
    while(1)
    {
        sleep(1);
    }
    return 0;
}

本身父进程就会忽略这个信号,那为什么还要这样写出来?如果这样写的话,会修改父进程pcb的状态位,子进程继承过去,然后按照这个办法就执行了,因为这相当于系统调用,会修改pcb。但是这个只在Linux中有效,不保证其他Unix系统上可用。

结束。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

博客系统 —— Servlet 实现(前后端分离)(详细+整体代码+效果图)

目录 一、项目效果展示 二、创建 Servlet 项目 三、编写数据库的操作代码 1、创建数据库/表结构&#xff08;数据库设计&#xff09;&#xff08;Model&#xff09; 2、封装数据库操作&#xff08;Model&#xff09; &#xff08;1&#xff09;先创建 DBUtil 封装数据库连…

Etsy总是不出单怎么办?出单后怎么操作?

如果您在Etsy上总是无法出单&#xff0c;可以尝试以下几个步骤&#xff1a; 1、检查您的商品列表是否符合Etsy的要求&#xff0c;包括商品描述、价格、运费等信息是否准确无误。 2、确认您的账户信息是否完整&#xff0c;包括银行账户、信用卡信息等。 3、检查您的店铺设置是否…

socket API的使用+client/server代码演示+封装socket模块

前言&#xff1a;本章旨在讲解常见的socket API及其使用练习&#xff0c;尾部有封装好的&#xff08;socket相关的一些系统函数加上错误处理代码包装成新的函数&#xff09;模块wrap.c 目录 一、socket模型创建流程图 1&#xff09;socket函数 2&#xff09;bind函数 3&am…

Altium软件中相同模块布局布线的方法

文章目录 1、原理图设计1、绘制xxx.SchDoc&#xff0c;并设置port。具体方法&#xff1a;Place→Port。2、新建顶层原理图&#xff1a;可以命名为xxx_TOP3、repeat 原理图&#xff0c;将这里从XXX_SingleDut 改为 Repeat(S,1,12)4、以总线的方式出线&#xff0c;如下&#xff1…

网络安全CVE 漏洞分析及复现

漏洞详情 Shiro 在路径控制的时候&#xff0c;未能对传入的 url 编码进行 decode 解码&#xff0c;导致攻击者可以绕过过滤器&#xff0c;访问被过滤的路径。 漏洞影响版本 Shiro 1.0.0-incubating 对应 Maven Repo 里面也有 【一一帮助安全学习&#xff0c;所有资源获取一一…

NetBackup 10.2 新功能介绍:PostgreSQL 和 MySQL 自动化恢复达成

NetBackup 10.2 新功能介绍&#xff1a;PostgreSQL 和 MySQL 自动化恢复达成 原文来自&#xff1a;VERITAS 中文社区 2023-04-27 在执行恢复任务时&#xff0c;手动提取、更新数据库和实例并将其附加到 PostgreSQL 和 MySQL 是常规操作。而在最新的 NetBackup 10.2 版本中&am…

数据可视化工具 - ECharts以及柱状图的编写

1 快速上手 引入echarts 插件文件到html页面中 <head><meta charset"utf-8"/><title>ECharts</title><!-- step1 引入刚刚下载的 ECharts 文件 --><script src"./echarts.js"></script> </head>准备一个…

apc-service-bus项目Docker镜像发布

apc-service-bus项目Docker镜像发布 1. 提交代码到Gitee代码仓&#xff0c;通过建木将项目打包到服务器 1.1 可直接打开访问建木&#xff0c;无有不熟悉建木发布流程的请咨询其他同事或者自行研究 建木地址&#xff1a;http://10.11.148.21/ 1.2 找到bus的开发环境部署执行…

神经网络全连接层数学推导

全连接层分析 对于神经网络为什么都能产生很好的效果虽然其是一个黑盒&#xff0c;但是我们也可以对其中的一些数学推导有一定的了解。 数学背景 目标函数为 f ∣ ∣ m a x ( X W , 0 ) − Y ∣ ∣ F 2 &#xff0c;求 ∂ f ∂ W , ∂ f ∂ X , ∂ f ∂ Y 目标函数为f ||ma…

SpringBoot项目如何打包成exe应用程序

准备 准备工作&#xff1a; 一个jar包&#xff0c;没有bug能正常启动的jar包 exe4j&#xff0c;一个将jar转换成exe的工具 链接: https://pan.baidu.com/s/1m1qA31Z8MEcWWkp9qe8AiA 提取码: f1wt inno setup&#xff0c;一个将依赖和exe一起打成一个安装程序的工具 链接:…

设计模式——代理模式(静态代理、JDK动态代理、CGLIB动态代理)

是什么&#xff1f; 如果因为某些原因访问对象不适合&#xff0c;或者不能直接引用目标对象&#xff0c;这个时候就需要给该对象提供一个代理以控制对该对象的访问&#xff0c;代理对象作为访问对象和目标对象之间的中介&#xff1b; Java中的代理按照代理类生成时机不同又分…

婴儿摇篮语音播放芯片,高品质MP3音乐播放芯片,WT2003H

婴儿摇篮是一种用于帮助婴儿入睡的设备。传统的婴儿摇篮通常只是简单的摇晃&#xff0c;但是带有语音播报芯片的婴儿摇篮则可以更好地模拟妈妈的声音&#xff0c;从而更有效地帮助婴儿入睡。 如果您正在寻找高品质音乐摇篮方案&#xff0c;那么WT2003H语音播放芯片&#xff0c…

5月7日 2H55min|5月8日8H50min|时间轴复盘|14:00~14:30

5月8日 todo list list4 40min ✅ |实际上用了50+50 list6 40min ✅ |实际上用了30+60 阅读+听力连做 100min ✅ 口语 day01 ✅ 口语 day02 口语 day03

6、并发事务控制MVCC汇总

1.并发事务控制 单版本控制-锁 先来看锁&#xff0c;锁用独占的方式来保证在只有一个版本的情况下事务之间相互隔离&#xff0c;所以锁可以理解为单版本控制。 在 MySQL 事务中&#xff0c;锁的实现与隔离级别有关系&#xff0c;在 RR&#xff08;Repeatable Read&#xff0…

vCenter Server 8.0U1 OVF:在 Fusion 和 Workstation 中快速部署 vCSA

vCenter Server 8.0U1 OVF&#xff1a;在 Fusion 和 Workstation 中快速部署 vCSA vCenter Server 8.0U1 系列更新 请访问原文链接&#xff1a;https://sysin.org/blog/vmware-vcenter-8-ovf/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&#x…

Win上通过Jconsole查看Java程序资源占用情况(教程总结,一篇就够了)

最近需要读取一个大文件&#xff0c;为了判断有没有读取到内存中&#xff0c;需要一个能查看jar包占用内存的工具&#xff0c;一顿面向百度后&#xff0c;发现了jdk自带的工具Jconsole&#xff0c;将教程分享给大家 一、介绍 JConsole 是一个内置 Java 性能分析器&#xff0c;…

手把手教你使用unisat 交易市场|BRC20|Unisat

开始前先熟悉下这张平台市场标注图&#xff0c;能让你跟得心应手&#xff01; 一、查看实时成交信息&#xff08;已 moon 为例子&#xff09; 搜索进入Token界面&#xff0c;点击 Orders 可以看到 1w 枚成交 87.86U&#xff08;单价 30 聪&#xff0c;大约 0.008786u&#xf…

牛客网剑指offer|中等题day2|JZ76删除链表中的重复节点、JZ23链表中环的入口节点、JZ24 反转链表(简单)

JZ76删除链表中的重复节点 链接&#xff1a;删除链表中重复的结点_牛客题霸_牛客网 参考代码&#xff1a; 自己好像还是偏向双指针这种想法&#xff0c;所以用了两个指针&#xff0c;这样感觉更好理解一些。 对了&#xff0c;去重有两种&#xff0c;我一开始写成了简单的那种&a…

MGV3001_ZG_当贝纯净桌面-线刷固件包

MGV3001_ZG_当贝纯净桌面-线刷固件包-内有教程及短接点 特点&#xff1a; 1、适用于对应型号的电视盒子刷机&#xff1b; 2、开放原厂固件屏蔽的市场安装和u盘安装apk&#xff1b; 3、修改dns&#xff0c;三网通用&#xff1b; 4、大量精简内置的没用的软件&#xff0c;运…

【标准化方法】(3) Group Normalization 原理解析、代码复现,附Pytorch代码

今天和各位分享一下深度学习中常用的标准化方法&#xff0c;Group Normalization 数据分组归一化&#xff0c;向大家介绍一下数学原理&#xff0c;并用 Pytorch 复现。 Group Normalization 论文地址&#xff1a;https://arxiv.org/pdf/1803.08494.pdf 1. 原理介绍 在目标检测…