C++学习---信号处理机制、中断、异步环境

前言


信号处理

关于信号,信号是一种进程间通信的机制,用于在程序执行过程中通知进程发生了一些事件。在Unix和类Unix系统中,信号是一种异步通知机制,通过发送信号,一个进程可以通知另一个进程发生了某个事件,如按下 Ctrl+C、除零错误等。

在C++中,可以使用 头文件提供的信号处理机制来捕获和处理信号。

信号的基本概念:

  • 信号编号:每个信号都有一个唯一的编号,用来标识不同的事件。例如,SIGINT 是表示中断的信号。
  • 信号处理器: 信号处理器是一个函数,用于处理接收到的信号。你可以为每种信号指定一个处理函数。

常见的信号

  • SIGINT:中断信号,通常由用户按下 Ctrl+C 生成。
  • SIGSEGV:段错误信号,表示非法内存访问。
  • SIGTERM:终止信号,表示进程被要求终止。
  • SIGKILL:强制终止信号,表示进程被强制终止。

signal()函数

C++使用 signal 函数可以为特定的信号注册信号处理函数

语法使用:

void (*signal(int signum, void (*handler)(int)))(int);

也可以写成:

signal(SIGINT, signalHandler);

在上述代码中,

  • signum(SIGINT):要注册的信号的编号。
  • handler(signalHandler):要注册的信号处理函数的指针。

信号处理函数的声明:

void handlerFunction(int signum)

关于信号处理函数的定义应该尽量简单,因为它在异步环境中执行,同时有一些函数(例如‘printf’,‘malloc’)不是异步安全的,所以尽量不要在信号处理函数中使用它们。

关于异步环境

异步环境是指程序执行时存在多个同时运行的线程或进程,这些线程或进程在执行过程中可能会相互干扰,因为它们共享某些资源(如内存、文件描述符等)。在异步环境中,执行顺序是不确定的,因此程序的行为可能受到非常复杂的影响。

printfmalloc 不是异步安全的主要是因为它们在执行时可能涉及到对共享资源的访问,而这样的访问在异步环境中是不安全的。

  1. printf:
    • print函数通常会使用标准输出(stdout),而在异步环境中,多个线程或进程可能会同时尝试写入标准输出,导致输出内容混乱。
    • 在标准库中的输出函数(如 printf)通常使用全局锁(mutex)来保护对输出流的访问,但这并不能解决所有的异步安全问题。在信号处理函数中使用 printf 可能导致死锁或其他竞态条件。
  2. malloc:
    • malloc 函数用于动态分配内存,而在异步环境中,多个线程或进程可能同时尝试分配或释放内存,这可能导致内存管理错误。

因此,在异步环境中,为了确保代码的正确性,应该尽量避免在信号处理函数或多线程环境中使用不可重入(non-reentrant)的函数。不可重入函数是指在执行过程中依赖于全局状态静态变量的函数,而这在异步环境中可能导致不确定的结果。

为了在异步环境中安全使用输出函数和内存分配函数,通常建议使用异步安全的替代版本。例如,在信号处理函数中,可以使用 write 函数代替 printf,而在多线程环境中,可以使用 pthread 库提供的线程安全的输出函数和内存分配函数。


信号处理函数示例

#include <iostream>
#include <csignal>

// 信号处理函数
void signalHandler(int signum) {
    std::cout << "Received signal: " << signum << std::endl;

    // 自定义处理逻辑可以在这里添加
    // ...

    // 恢复对 SIGINT 的默认处理
    signal(SIGINT, SIG_DFL);
}

int main() {
    // 注册信号处理函数
    signal(SIGINT, signalHandler);

    std::cout << "Press Ctrl+C to trigger the signal." << std::endl;

    // 一个简单的循环,使程序保持运行
    while (true) {
        // 等待信号的到来
    }

    return 0;
}

在上述代码中,声明了一个自定义的信号处理函数signalHandler;之后的main函数中,用signal注册信号处理函数,来处理SIGINT信号。signal函数的第一个参数就是要识别的信号的编号,第二个参数就是指向信号处理函数的指针。

而在信号处理函数signalHandler中,可以添加自定义的处理逻辑。

上述代码运行后,因为while(ture),程序会一直保持运行,直到我们按下Ctrl+C发生中断后,signal函数在捕获信号后,信号处理函数发挥作用,打印了Received signal: 2;

因为SIGINT 的信号编号是 2,所以signum的值是2;

如果想要在信号处理完成后恢复对该信号的默认处理,可以使用 signal(SIGINT, SIG_DFL)。

忽略和恢复信号:

  • 使用 signal(SIGINT, SIG_IGN) 可以忽略 SIGINT 信号。
  • 使用 signal(SIGINT, SIG_DFL) 可以恢复对 SIGINT 的默认处理。

关于恢复信号,当你按下 Ctrl+C 触发 SIGINT 信号时,如果没有 signal(SIGINT, SIG_DFL); 这一行,那么程序将继续执行 signalHandler 函数,但不会将 SIGINT 的处理方式恢复为默认。这意味着如果再次按下 Ctrl+C,signalHandler 函数将再次被调用,而不会终止程序。

实际上,如果不将 SIGINT 恢复为默认处理方式,程序可能会对多次 Ctrl+C 信号作出相应,而不是默认的行为(终止程序)。


raise()函数

raise 函数是用于在程序中手动触发一个信号的函数。

声明:

int raise(int sig);
  • sig:要触发的信号的编号。

raise 函数返回一个整数值,表示函数调用的结果。如果成功发送信号,返回 0;如果失败,返回非零值。

示例:

#include <csignal>
#include <iostream>


// 信号处理函数
void signalHandler(int signum) {
    std::cout << "Received signal: " << signum << std::endl;
}

int main() {
    // 注册信号处理函数
    signal(SIGINT, signalHandler);

    std::cout << "Press Ctrl+C to trigger the signal." << std::endl;

    // 模拟其他程序逻辑
    int count = 0;
    while (true) {
        // 模拟其他程序逻辑
        std::cout << "Working... (" << count << ")" << std::endl;

        // 在某个条件下手动触发 SIGINT 信号
        if (count > 500) {
            std::cout << "Manually triggering SIGINT..." << std::endl;
            raise(SIGINT);
        }

        // 模拟其他程序逻辑
        // ...
        

        // 增加计数
        count++;
    }

    return 0;
}

上述代码的运行结果:

在这里插入图片描述

可以看到在进行俩次的模拟生成信号后,程序就停止了,这是因为没有重新注册 SIGINT 的处理函数,程序将使用默认的信号处理方式,即终止程序。

如果想要第一次信号处理后继续运行,可以重新注册 SIGINT 的处理函数。
在signalHandler函数中添加:

// 重新注册信号处理函数
	signal(SIGINT, signalHandler);
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在Dev C++中,断点是一种调试工具,它可以让程序在执行到指定的代码行时停下来,以便程序员检查程序的状态和变量的值。通过设置断点,程序员可以逐行调试程序,找出程序中的错误和问题。在调试过程中,程序员可以查看变量的值、修改变量的值、单步执行程序、跳过某些代码行等操作,以便更好地理解程序的执行过程和调试程序。\n\举个例子,假设我们在调试一个程序时,发现程序在某个地方出现了错误,但是我们不知道具体是哪一行代码出现了问题。这时,我们可以在程序中设置一个断点,让程序在执行到这一行代码时停下来,然后逐行检查程序的状态和变量的值,找出问题所在。这样,我们就可以更快地定位和解决程序中的错误。\n\下面是在Dev C++中设置断点的基本方法和过程:\1. 打开需要调试的源代码文件,找到需要设置断点的代码行。\2. 在代码行的左侧单击鼠标左键,或者按下F9键,即可在该代码行上设置一个断点。设置成功后,该代码行的左侧会出现一个红色的圆点,表示该行代码已经设置了断点。\3. 在程序运行时,当程序执行到设置的断点处时,程序会自动停下来,此时可以查看程序的状态和变量的值,进行调试操作。\4. 在调试过程中,可以使用F8键单步执行程序,使用F7键进入函数调用,使用F4键查看变量的值,使用F5键继续执行程序,使用Shif+F5键停止程序的执行。\n\

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Sciurdae.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值