不同进程间发送与接收携带字符串的信号(使用共享内存方式)


在学习的过程中发现,在父子进程,或者同一进程中发送一个带有字符串的信号是可以完成字符串打印的工作,但是如果在不同进程之间打印传递的字符串便会出现段错误的情况,为了解决这个问题,可以使用共享内存的方法,即在发送信号时,将我们需要传递的字符串放入共享内存中,在接收信号时,再将我们传递的数据从共享内存中将字符串读取出来。

sig_receve:接收端

#include <stdio.h>
#include <signal.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>

// 定义处理 SIGUSR1 信号的函数
void handler_USR1(int signum, siginfo_t *info, void *content) {
    
    printf("get signum %d\n", signum);// 打印接收到的信号编号
    int data = info->si_int;  //从siginfo_t结构体中获取附加的整数值
    
    // 如果 content 不为空,则打印附加的整数值
    if (content != NULL) {
        printf("get data = %d\n", data);
    }
}

// 定义处理 SIGUSR2 信号的函数
void handler_USR2(int signum, siginfo_t *info, void *content) {
    
    printf("get signum %d\n", signum);// 打印接收到的信号编号
    int shmid;
    char *shmaddr = info->si_ptr;// 从 siginfo_t 结构体中获取附加的共享内存地址
    
    key_t key;// 用于创建共享内存键的 key_t 类型变量
    key = ftok(".", 3);// 创建一个唯一的键
    shmid = shmget(key, 1024 * 4, 0);// 使用 ftok 生成的键来获取共享内存的标识符
    shmaddr = shmat(shmid, NULL, 0);// 将共享内存附加到调用进程的地址空间
    
    // 如果 content 不为空,则打印共享内存中的字符串
    if (content != NULL) {
        printf("str = %s\n", shmaddr);
    }
    
    shmdt(shmaddr);// 从进程的地址空间分离共享内存
    shmctl(shmid, IPC_RMID, 0); // 删除共享内存段
}

int main() {
    // 定义 sigaction 结构体,为两个信号设置信号处理函数
    struct sigaction act1, act2;
    act1.sa_sigaction = handler_USR1; // 设置 SIGUSR1 的信号处理函数
    act1.sa_flags = SA_SIGINFO;       // 设置标志,使用 siginfo_t 结构体传递信号信息

    act2.sa_sigaction = handler_USR2; // 设置 SIGUSR2 的信号处理函数
    act2.sa_flags = SA_SIGINFO;       // 设置标志,使用 siginfo_t 结构体传递信号信息

    
    printf("pid = %d\n", getpid());// 打印当前进程的 PID
   
    sigemptyset(&act1.sa_mask); // 初始化信号掩码
    // 为 SIGUSR1 信号设置信号处理函数
    if (sigaction(SIGUSR1, &act1, NULL) == -1) {
        perror("sigaction1");
    }
    sigemptyset(&act2.sa_mask);
    
    // 为 SIGUSR2 信号设置信号处理函数
    if (sigaction(SIGUSR2, &act2, NULL) == -1) {
        perror("sigaction2");
    }
    
    
    while (1);// 一直等待信号触发
    return 0;
}

sig_send:发送端
 

#include <stdio.h>
#include <signal.h>
#include <string.h>
#include <sys/shm.h>
#include <sys/ipc.h>


int main(int argc, char **argv) {
    // 检查命令行参数数量是否正确
    if (argc != 3) {
        printf("param error\n");// 如果参数数量不正确,打印错误信息并退出程序
        exit(EXIT_FAILURE);
    }
    
    int pid = atoi(argv[1]);// 从命令行参数中获取目标进程的 PID
    int signum = atoi(argv[2]);// 从命令行参数中获取信号编号

    key_t key;
    key = ftok(".", 3);// 使用 ftok 生成共享内存的键

    
    union sigval value1, value2;// 定义两个 sigval 联合体,用于存储要随信号发送的附加数据
    value1.sival_int = 98; // 设置 value1 为一个整数 98
    
    // 如果信号编号为 SIGUSR1,则向目标进程发送信号和附加的整数值
    if (signum == SIGUSR1) { 
        if (sigqueue(pid, signum, value1) == -1) {
            perror("sigqueue1");
        }
        printf("SIGUSR1 send ok\n");
    }

    
    int shmid = shmget(key, 1024 * 4, IPC_CREAT | 0666);// 创建或获取共享内存的标识
    char *shmaddr = shmat(shmid, NULL, 0);// 将共享内存附加到当前进程的地址空间 
    strcpy(shmaddr, "hello word");// 向共享内存中复制字符串 "hello word"
    value2.sival_ptr = shmaddr; // 设置 value2 为共享内存的地址
    
    // 如果信号编号为 SIGUSR2,则向目标进程发送信号和附加的共享内存地址
    if (signum == SIGUSR2) { 
        if (sigqueue(pid, signum, value2) == -1) {
            perror("sigqueue2");
        }
        printf("SIGUSR2 send ok\n");
    }

    return 0;
}

运行结果:左侧接收信号,并做出相应的处理,右侧发送信号。


信号处理函数应该尽量简单和快速,避免在其中进行耗时或阻塞的操作,以免影响程序的正常执行。对于复杂的信号处理逻辑,可能需要考虑使用其他机制,如信号队列或异步 I/O。

参考链接:http://t.csdnimg.cn/q5JuH
 

  • 4
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值