900字范文,内容丰富有趣,生活中的好帮手!
900字范文 > linux signal函数使用

linux signal函数使用

时间:2022-11-09 07:27:32

相关推荐

linux signal函数使用

signal()函数理解

在<signal.h> 这个头文件中。

signal(参数1,参数2);

参数1:我们要进行处理的信号。系统的信号我们可以再终端键入 kill -l查看(共64个)。其实这些信号时系统定义的宏。

参数2:我们处理的方式(是系统默认还是忽略还是捕获)。

一般有3中方式进行操作。

(1)eg: signal(SIGINT ,SIG_ING );

//SIG_ING 代表忽略SIGINT信号,SIGINT信号代表由InterruptKey产生,通常是CTRL +C 或者是DELETE 。发送给所有ForeGround Group的进程。

下面我们写个死循环:

这时我们保存执行。

按下CTRL _C程序没有反应。这就对了

如果我们想结束该程序可以按下CTRL +\来结束

其实当我们按下CTRL +\组合键时,是产生了SIGQUIT信号

(2)eg: signal(SIGINT ,SIG_DFL );

//SIGINT信号代表由InterruptKey产生,通常是CTRL +C或者是DELETE。发送给所有ForeGroundGroup的进程。 SIG_DFL代表执行系统默认操作,其实对于大多数信号的系统默认动作时终止该进程。这与不写此处理函数是一样的。

我们将上面的程序改成

这时就可以按下CTRL +C 来终止该进程了。把signal(SIGINT,SIG_DFL);这句去掉,效果是一样的。

(3) void ( *signal( int sig, void (* handler)( int )))( int );

int (*p)();

这是一个函数指针, p所指向的函数是一个不带任何参数, 并且返回值为int的一个函数.

int (*fun())();

这个式子与上面式子的区别在于用fun()代替了p,而fun()是一个函数,所以说就可以看成是fun()这个函数执行之后,它的返回值是一个函数指针,这个函数指针(其实就是上面的p)所指向的函数是一个不带任何参数,并且返回值为int的一个函数.

void (*signal(int signo, void (*handler)(int)))(int);就可以看成是signal()函数(它自己是带两个参数,一个为整型,一个为函数指针的函数),而这个signal()函数的返回值也为一个函数指针,这个函数指针指向一个带一个整型参数,并且返回值为void的一个函数.

在写信号处理函数时对于信号处理的函数也是void sig_fun(int signo);这种类型,恰好与上面signal()函数所返回的函数指针所指向的函数是一样的.void ( *signal() )( int );

signal是一个函数, 它返回一个函数指针, 后者所指向的函数接受一个整型参数 且没有返回值, 仔细看, 是不是siganal( int signo, void (*handler)(int) )的第2个参数了,对了,其实他所返回的就是 signal的第2个信号处理函数,指向信号处理函数,就可以执行函数了( signal内部时, signal把信号做为参数传递给handler信号处理函数,接着 signal函数返回指针,并且又指向信号处理函数,就开始执行它)

那么,signal函数的参数又是如何呢?signal函数接受两个参数:一个整型的信号编号,以及一个指向用户定义的信号处理函数的指针。我们此前已经定义了指向用户定义的信号处理函数的指针sfp:

sfp 的类型可以通过将上面的声明中的sfp去掉而得到,即void (*)(int)。此外,signal函数的返回值是一个指向调用前的用户定义信号处理函数的指针,这个指针的类型与sfp指针类型一致。因此,我们可以如下声明signal函数:

同样地,使用typedef可以简化上面的函数声明:

Ok;看个例子:

******************************************************************************************************************************

1.

int (*func)();函数指针,指向的函数为空参数,返回整型;

2.

回调函数是一个程序员不能显式调用的函数;通过将回调函数的地址传给被调用者从而实现调用。

回调函数是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数。

void f();// 函数原型

上面的语句声明了一个函数,没有输入参数并返回void。那么函数指针的声明方法如下:

void (*) ();

让我们来分析一下,左边圆括弧中的星号是函数指针声明的关键。另外两个元素是函数的返回类型(void)和由边圆括弧中的入口参数(本例中参数是空)。注意本例中还没有创建指针变量-只是声明了变量类型。目前可以用这个变量类型来创建类型定义名及用sizeof表达式获得函数指针的大小:

// 获得函数指针的大小

unsigned psize = sizeof (void (*) ());

// 为函数指针声明类型定义

typedef void (*pfv) ();

pfv是一个函数指针,它指向的函数没有输入参数,返回类行为void。使用这个类型定义名可以隐藏复杂的函数指针语法。

指针变量应该有一个变量名:

void (*p) (); //p是指向某函数的指针

p是指向某函数的指针,该函数无输入参数,返回值的类型为void。左边圆括弧里星号后的就是指针变量名。有了指针变量便可以赋值,值的内容是署名匹配的函数名和返回类型。例如:

void func()

{

/* do something */

}

p = func;

p的赋值可以不同,但一定要是函数的地址,并且署名和返回类型相同。

传递回调函数的地址给调用者

现在可以将p传递给另一个函数(调用者)- caller(),它将调用p指向的函数,而此函数名是未知的:

void caller(void(*ptr)())

{

ptr(); /* 调用ptr指向的函数 */

}

void func();

int main()

{

p = func;

caller(p); /* 传递函数地址到调用者 */

}

如果赋了不同的值给p(不同函数地址),那么调用者将调用不同地址的函数。赋值可以发生在运行时,这样使你能实现动态绑定。

3.函数指针在linux中的应用signal函数

在Unix/Linux中signal函数是比较复杂的一个,其定义原型如下: void (*signal(int signo,void (*func)(int))) (int) signal(int signo, void(*func)(int))是signal函数的主体. 需要两个参数:int型的signo,以及一个指向函数的指针. void (*func)(int). 这个函数中,最外层的函数体 void (* XXX )(int)表明其返回值是一个指针(函数指针),指向一个函数XXX的指针,XXX所代表的函数需要一个int型的参数,返回void。 正是由于其复杂性,在[Plauger 1992]用typedef来对其进行简化 typedef void Sigfuc(int);//这里可以看成一个返回值 . 再对signal函数进行简化就是这样的了 Sigfunc *signal(int,Sigfuc *); 在signal.h头文件中还有以下几个定义 #define SIG_ERR (void (*)())-1 #define SIG_DFL (void (*)())0

#define SIG_IGN (void (*)())1

alarm(设置信号传送闹钟)

相关函数 signal,sleep

表头文件 #include

定义函数 unsigned int alarm(unsigned int seconds);

函数说明 alarm()用来设置信号SIGALRM在经过参数seconds指定的秒数后传送给目前的进程。如果参数seconds 为0,则之前设置的闹钟会被取消,并将剩下的时间返回。

返回值返回之前闹钟的剩余秒数,如果之前未设闹钟则返回0。

范例 #include

#include

void handler() {

printf("hello\n");

}

main()

{

int i;

signal(SIGALRM,handler);

alarm(5);

for(i=1;i<7;i++){

printf("sleep %d ...\n",i);

sleep(1);

}

}

执行 sleep 1 ...

sleep 2 ...

sleep 3 ...

sleep 4 ...

sleep 5 ...

hello

sleep 6 ...

kill(传送信号给指定的进程)

相关函数 raise,signal

表头文件 #include

#include

定义函数 int kill(pid_t pid,int sig);

函数说明 kill()可以用来送参数sig指定的信号给参数pid指定的进程。参数pid有几种情况:

pid>0 将信号传给进程识别码为pid 的进程。

pid=0 将信号传给和目前进程相同进程组的所有进程

pid=-1 将信号广播传送给系统内所有的进程

pid<0 将信号传给进程组识别码为pid绝对值的所有进程

参数sig代表的信号编号可参考附录D

返回值 执行成功则返回0,如果有错误则返回-1。

错误代码 EINVAL 参数sig 不合法

ESRCH 参数pid 所指定的进程或进程组不存在

EPERM 权限不够无法传送信号给指定进程

范例 #include

#include

#include

#include

main()

{

pid_t pid;

int status;

if(!(pid= fork())){

printf("Hi I am child process!\n");

sleep(10);

return;

}

else{

printf("send signal to child process (%d) \n",pid);

sleep(1);

kill(pid ,SIGABRT);

wait(&status);

if(WIFSIGNALED(status))

printf("chile process receive signal %d\n",WTERMSIG(status));

}

}

执行 sen signal to child process(3170)

Hi I am child process!

child process receive signal 6

pause(让进程暂停直到信号出现)

相关函数 kill,signal,sleep

表头文件 #include

定义函数 int pause(void);

函数说明 pause()会令目前的进程暂停(进入睡眠状态),直到被信号(signal)所中断。

返回值 只返回-1。

错误代码 EINTR 有信号到达中断了此函数。

sigaction(查询或设置信号处理方式)

相关函数 signal,sigprocmask,sigpending,sigsuspend

表头文件 #include

定义函数 int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact);

函数说明 sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。

如参数结构sigaction定义如下

struct sigaction

{

void (*sa_handler) (int);

sigset_t sa_mask;

int sa_flags;

void (*sa_restorer) (void);

}

sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。

sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。

sa_restorer 此参数没有使用。

sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。

OR 运算(|)组合

A_NOCLDSTOP : 如果参数signum为SIGCHLD,则当子进程暂停时并不会通知父进程

SA_ONESHOT/SA_RESETHAND:当调用新的信号处理函数前,将此信号处理方式改为系统预设的方式。

SA_RESTART:被信号中断的系统调用会自行重启

SA_NOMASK/SA_NODEFER:在处理此信号未结束前不理会此信号的再次到来。

如果参数oldact不是NULL指针,则原来的信号处理方式会由此结构sigaction 返回。

返回值 执行成功则返回0,如果有错误则返回-1。

错误代码 EINVAL 参数signum 不合法,或是企图拦截SIGKILL/SIGSTOPSIGKILL信号

EFAULT 参数act,oldact指针地址无法存取。

EINTR 此调用被中断

范例 #include

#include

void show_handler(struct sigaction * act)

{

switch (act->sa_flags)

{

case SIG_DFL:printf("Default action\n");break;

case SIG_IGN:printf("Ignore the signal\n");break;

default: printf("0x%x\n",act->sa_handler);

}

}

main()

{

int i;

struct sigaction act,oldact;

act.sa_handler = show_handler;

act.sa_flags = SA_ONESHOT|SA_NOMASK;

sigaction(SIGUSR1,&act,&oldact);

for(i=5;i<15;i++)

{

printf("sa_handler of signal %2d =".i);

sigaction(i,NULL,&oldact);

}

}

执行 sa_handler of signal 5 = Default action

sa_handler of signal 6= Default action

sa_handler of signal 7 = Default action

sa_handler of signal 8 = Default action

sa_handler of signal 9 = Default action

sa_handler of signal 10 = 0x8048400

sa_handler of signal 11 = Default action

sa_handler of signal 12 = Default action

sa_handler of signal 13 = Default action

sa_handler of signal 14 = Default action

sigaddset(增加一个信号至信号集)

相关函数 sigemptyset,sigfillset,sigdelset,sigismember

表头文件 #include

定义函数 int sigaddset(sigset_t *set,int signum);

函数说明 sigaddset()用来将参数signum 代表的信号加入至参数set 信号集里。

返回值执行成功则返回0,如果有错误则返回-1。

错误代码 EFAULT 参数set指针地址无法存取

EINVAL 参数signum非合法的信号编号

sigdelset(从信号集里删除一个信号)

相关函数 sigemptyset,sigfillset,sigaddset,sigismember

表头文件 #include

定义函数 int sigdelset(sigset_t * set,int signum);

函数说明 sigdelset()用来将参数signum代表的信号从参数set信号集里删除。

返回值 执行成功则返回0,如果有错误则返回-1。

错误代码 EFAULT 参数set指针地址无法存取

EINVAL 参数signum非合法的信号编号

sigemptyset(初始化信号集)

相关函数 sigaddset,sigfillset,sigdelset,sigismember

表头文件 #include

定义函数 int sigemptyset(sigset_t *set);

函数说明 sigemptyset()用来将参数set信号集初始化并清空。

返回值 执行成功则返回0,如果有错误则返回-1。

错误代码 EFAULT 参数set指针地址无法存取

sigfillset(将所有信号加入至信号集)

相关函数 sigempty,sigaddset,sigdelset,sigismember

表头文件 #include

定义函数 int sigfillset(sigset_t * set);

函数说明 sigfillset()用来将参数set信号集初始化,然后把所有的信号加入到此信号集里。

返回值 执行成功则返回0,如果有错误则返回-1。

附加说明 EFAULT 参数set指针地址无法存取

sigismember(测试某个信号是否已加入至信号集里)

相关函数 sigemptyset,sigfillset,sigaddset,sigdelset

表头文件 #include

定义函数 int sigismember(const sigset_t *set,int signum);

函数说明 sigismember()用来测试参数signum 代表的信号是否已加入至参数set信号集里。如果信号集里已有该信号则返回1,否则返回0。

返回值信号集已有该信号则返回1,没有则返回0。如果有错误则返回-1。

错误代码 EFAULT 参数set指针地址无法存取

EINVAL 参数signum 非合法的信号编号

signal(设置信号处理方式)

相关函数 sigaction,kill,raise

表头文件 #include

定义函数 void (*signal(int signum,void(* handler)(int)))(int);

函数说明 signal()会依参数signum 指定的信号编号来设置该信号的处理函数。当指定的信号到达时就会跳转到参数handler指定的函数执行。如果参数handler不是函数指针,则必须是下列两个常数之一:

SIG_IGN 忽略参数signum指定的信号。

SIG_DFL 将参数signum 指定的信号重设为核心预设的信号处理方式。

关于信号的编号和说明,请参考附录D

返回值返回先前的信号处理函数指针,如果有错误则返回SIG_ERR(-1)。

附加说明在信号发生跳转到自定的handler处理函数执行后,系统会自动将此处理函数换回原来系统预设的处理方式,如果要改变此操作请改用sigaction()。

范例 参考alarm()或raise()。

sigpending(查询被搁置的信号)

相关函数 signal,sigaction,sigprocmask,sigsuspend

表头文件 #include

定义函数 int sigpending(sigset_t *set);

函数说明 sigpending()会将被搁置的信号集合由参数set指针返回。

返回值执 行成功则返回0,如果有错误则返回-1。

错误代码 EFAULT 参数set指针地址无法存取

EINTR 此调用被中断。

sigprocmask(查询或设置信号遮罩)

相关函数 signal,sigaction,sigpending,sigsuspend

表头文件 #include

定义函数 int sigprocmask(int how,const sigset_t *set,sigset_t * oldset);

函数说明 sigprocmask()可以用来改变目前的信号遮罩,其操作依参数how来决定

SIG_BLOCK 新的信号遮罩由目前的信号遮罩和参数set 指定的信号遮罩作联集

SIG_UNBLOCK 将目前的信号遮罩删除掉参数set指定的信号遮罩

SIG_SETMASK 将目前的信号遮罩设成参数set指定的信号遮罩。

如果参数oldset不是NULL指针,那么目前的信号遮罩会由此指针返回。

返回值 执行成功则返回0,如果有错误则返回-1。

错误代码 EFAULT 参数set,oldset指针地址无法存取。

EINTR 此调用被中断

sleep(让进程暂停执行一段时间)

相关函数 signal,alarm

表头文件 #include

定义函数 unsigned int sleep(unsigned int seconds);

函数说明 sleep()会令目前的进程暂停,直到达到参数seconds 所指定的时间,或是被信号所中断。

返回值 若进程暂停到参数seconds 所指定的时间则返回0,若有信号中断则返回剩余秒数

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。