UNIX(编程-信号处理):17---信号发送函数(sigqueue)

一、sigqueue函数

#include <signal.h>
int sigqueue(pid_t pid, int signo, const union sigval value);

 参数:

  • 参数1:信息传递给哪个进程
  • 参数2:要穿传递给参数1进程的信号
  • 参数3:信号附加信息,为一个union sigval联合体,包括一个sival_int整型和一个sival_ptr指针

返回值:

  • 成功,返回0;出错,返回-1

功能:

  • 此函数与sigaction信号捕获函数配合使用。此函数将参数2的信号传递给参数1所指向的进程,并且将参数3的附加信息也传递给该基金成
  • 类似于kill函数

如何与sigaction配合使用?

  • 第一步:sigqueue函数将参数2的信号和参数3的附加信息发送给参数1所指向的进程
  • 第二步:参数1所指向的进程如果使用sigaction函数捕获该信号,并且参数2的struct sigaction结构体采用flags=SA_SIGINFO和sa_action函数处理该信号
  • 第三步:承上,则sigqueue参数3的联合体信息就会传递给sa_action函数参数2的siginfo_t结构体中的si_value成员,该si_value成员的sival_int和sival_ptr就是sigqueue参数3的内容

注意事项:

UNIX(编程-信号处理):17---信号发送函数(sigqueue)

UNIX(编程-信号处理):17---信号发送函数(sigqueue)

二、应用

  • siginfo_t的应用:有时我们接收到一个信号时,希望接受一些其他数据,就可以设置这个成员
  • 例如:我们有一个用链表实现一个简单的服务器,并且想要在规定的一个时间间隔发送一些数据。就可以定一个定时器,并且在接收到SIGALRM信号时,因为操作数据需要用到链表,将链表地址传递给union sigval的sival_ptr指针。这样当捕获SIGALRM信号时,就可以调用sa_action函数,然后通过参数2的sival_ptr获取这个指针

演示案例:

  • 我们用一个进程给另外一个进程发送SIGINT信号,并且同时将一个整型100发送出去
//发送进程:向一个进程发送SIGINT信号,并且将整型100也传进去
#include<stdio.h>
#include<signal.h>
#include<unistd.h>
#include<string.h>

int main(int argc,char *argv[])
{
    if(argc!=2)
    {
        printf("arguments error!");
        exit(0);
    }

    pid_t pid=atoi(argv[1]);//将进程号转化为整数
    union sigval v;
    v.sival_int=100; //初始化sival_int的值

    sigqueue(pid,SIGINT,v);//向pid发送SIGINT信号,并且传递额外信息v
    return 0;
}
//此进程等待接受信号的传入,并有SIGINT的信号处理函数
#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<string.h>

void handler(int,siginfo_t *,void *);

int main(void)
{
    struct sigaction act;
    act.sa_sigaction=handler;
    sigemptyset(&act.sa_mask);
    act.sa_flags=SA_SIGINFO;
    if(sigaction(SIGINT,&act,NULL)<0)
    {
        printf("error");
        exit(0);
    }
    for(;;)
        pause();
    return 0;
}

void handler(int sig,siginfo_t * info,void *ctx)
{
    //打印信息
    printf("recv a sid=%d data=%d data=%d\n",sig,info->si_value.sival_int,info->si_int);
}
  • 演示结果:

UNIX(编程-信号处理):17---信号发送函数(sigqueue)