Linux编程: 在业务线程中注册和处理Linux信号

简介: 本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。

Linux编程:在业务线程中注册和处理Linux信号

在Linux编程中,信号是一种异步通知机制,用于通知进程某个事件的发生。常见的信号包括SIGINT、SIGTERM、SIGKILL等。处理信号通常在主线程中进行,但有时需要在业务线程中处理信号。本文将详细介绍如何在业务线程中注册和处理Linux信号。

一、Linux信号简介

1.1 信号的类型

信号分为两类:

  • 标准信号:如SIGINT、SIGTERM、SIGKILL等。
  • 实时信号:用于用户自定义信号,范围为SIGRTMIN到SIGRTMAX。

1.2 信号处理

信号处理包括以下几种方式:

  • 忽略信号:使用 SIG_IGN忽略信号。
  • 默认处理:使用 SIG_DFL进行默认处理。
  • 自定义处理:使用信号处理函数自定义处理逻辑。

二、在业务线程中注册信号处理

2.1 创建业务线程

使用 pthread库创建业务线程:

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

void* business_thread(void* arg) {
    // 业务线程的逻辑
    while (1) {
        printf("Business thread running...\n");
        sleep(1);
    }
    return NULL;
}

int main() {
    pthread_t thread;
    if (pthread_create(&thread, NULL, business_thread, NULL) != 0) {
        perror("Failed to create thread");
        return 1;
    }

    // 等待业务线程结束
    pthread_join(thread, NULL);
    return 0;
}
​

2.2 注册信号处理函数

在业务线程中注册信号处理函数:

void signal_handler(int sig) {
    printf("Received signal %d in business thread\n", sig);
}

void* business_thread(void* arg) {
    // 注册信号处理函数
    struct sigaction sa;
    sa.sa_handler = signal_handler;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;

    if (sigaction(SIGINT, &sa, NULL) == -1) {
        perror("Failed to register signal handler");
        pthread_exit(NULL);
    }

    // 业务线程的逻辑
    while (1) {
        printf("Business thread running...\n");
        sleep(1);
    }
    return NULL;
}
​

2.3 阻塞和解除阻塞信号

在创建业务线程之前,阻塞主线程中的信号,在业务线程中解除阻塞信号:

int main() {
    pthread_t thread;

    // 阻塞SIGINT信号
    sigset_t set;
    sigemptyset(&set);
    sigaddset(&set, SIGINT);
    pthread_sigmask(SIG_BLOCK, &set, NULL);

    if (pthread_create(&thread, NULL, business_thread, NULL) != 0) {
        perror("Failed to create thread");
        return 1;
    }

    // 等待业务线程结束
    pthread_join(thread, NULL);
    return 0;
}

void* business_thread(void* arg) {
    // 解除阻塞SIGINT信号
    sigset_t set;
    sigemptyset(&set);
    sigaddset(&set, SIGINT);
    pthread_sigmask(SIG_UNBLOCK, &set, NULL);

    // 注册信号处理函数
    struct sigaction sa;
    sa.sa_handler = signal_handler;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;

    if (sigaction(SIGINT, &sa, NULL) == -1) {
        perror("Failed to register signal handler");
        pthread_exit(NULL);
    }

    // 业务线程的逻辑
    while (1) {
        printf("Business thread running...\n");
        sleep(1);
    }
    return NULL;
}
​

2.4 完整示例

以下是一个完整的示例,展示了如何在业务线程中注册和处理信号:

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

// 信号处理函数
void signal_handler(int sig) {
    printf("Received signal %d in business thread\n", sig);
}

// 业务线程函数
void* business_thread(void* arg) {
    // 解除阻塞SIGINT信号
    sigset_t set;
    sigemptyset(&set);
    sigaddset(&set, SIGINT);
    pthread_sigmask(SIG_UNBLOCK, &set, NULL);

    // 注册信号处理函数
    struct sigaction sa;
    sa.sa_handler = signal_handler;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;

    if (sigaction(SIGINT, &sa, NULL) == -1) {
        perror("Failed to register signal handler");
        pthread_exit(NULL);
    }

    // 业务线程的逻辑
    while (1) {
        printf("Business thread running...\n");
        sleep(1);
    }
    return NULL;
}

int main() {
    pthread_t thread;

    // 阻塞SIGINT信号
    sigset_t set;
    sigemptyset(&set);
    sigaddset(&set, SIGINT);
    pthread_sigmask(SIG_BLOCK, &set, NULL);

    if (pthread_create(&thread, NULL, business_thread, NULL) != 0) {
        perror("Failed to create thread");
        return 1;
    }

    // 等待业务线程结束
    pthread_join(thread, NULL);
    return 0;
}
​

2.5 运行结果

编译并运行上述代码:

$ gcc -o signal_thread signal_thread.c -pthread
$ ./signal_thread
Business thread running...
Business thread running...
# 在此时按下Ctrl+C发送SIGINT信号
Received signal 2 in business thread
Business thread running...
​

可以看到,SIGINT信号在业务线程中被捕获和处理。

三、总结

本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。

目录
相关文章
|
2月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
181 6
|
5月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
318 83
|
2月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
306 0
|
3月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
278 16
|
7月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
286 0
|
10月前
|
存储 Linux API
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
在计算机系统的底层架构中,操作系统肩负着资源管理与任务调度的重任。当我们启动各类应用程序时,其背后复杂的运作机制便悄然展开。程序,作为静态的指令集合,如何在系统中实现动态执行?本文带你一探究竟!
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
|
9月前
|
安全 Linux
【Linux】阻塞信号|信号原理
本教程从信号的基本概念入手,逐步讲解了阻塞信号的实现方法及其应用场景。通过对这些技术的掌握,您可以更好地控制进程在处理信号时的行为,确保应用程序在复杂的多任务环境中正常运行。
328 84
|
8月前
|
并行计算 Linux
Linux内核中的线程和进程实现详解
了解进程和线程如何工作,可以帮助我们更好地编写程序,充分利用多核CPU,实现并行计算,提高系统的响应速度和计算效能。记住,适当平衡进程和线程的使用,既要拥有独立空间的'兄弟',也需要在'家庭'中分享和并行的成员。对于这个世界,现在,你应该有一个全新的认识。
308 67
|
7月前
|
安全 算法 Ubuntu
Linux(openssl)环境:编程控制让证书自签的技巧。
总结:在Linux环境中,OpenSSL是一个非常实用的工具,可以帮助我们轻松地生成自签名证书。通过上述三个简单步骤,即可为内部网络、测试环境或开发环境创建自签名证书。但在公共访问场景下,建议购买经过权威认证机构签发的证书,以避免安全警告。
337 13
|
9月前
|
JavaScript Ubuntu Linux
如何在阿里云的linux上搭建Node.js编程环境?
本指南介绍如何在阿里云Linux服务器(Ubuntu/CentOS)上搭建Node.js环境,包含两种安装方式:包管理器快速安装和NVM多版本管理。同时覆盖全局npm工具配置、应用部署示例(如Express服务)、PM2持久化运行、阿里云安全组设置及外部访问验证等步骤,助你完成开发与生产环境的搭建。