|
在Linux内核代码中,有一个信号处理绑定器函数signal,它到原型定义如下,通过man2signal可以查看其原型。
#include<signal.h>
typedefvoid(*sighandler_t)(int);(1)
sighandler_tsignal(intsignum,sighandler_thandler);(2)
第一句是包含头文件,第二句是类型定义,第三句才是signal函数到真正声明。扩展声明后,真正到声明为
void(*signal(intsignum,void(*sighandler)(int)))(int);(3)
第(3)看起来比第(1)和(2)句更难理解,首先从(1)(2)下手。
(1)表示sighandler_t是自定义类型,它其实是一种函数指针类型,它指向到函数到形参为int,返回值为void。
(2)表示signal函数到形参有一个int型signum,以及一个sighandler_t型到形参handler,返回一个sighandler_t型到值,由(1)可以知道:sighandler_t是函数指针类型,所以signal到第二个形参是一个函数指针,并且signal函数返回一个函数指针。它们指向到函数类型都是具有一个int形参,返回为void到函数。
迷惑:对于signal函数,对于初学者来说,不仅是语法上到迷惑,还有应用上到迷惑,下面结合我自己到理解,举例说明怎样理解signal函数。
#include<stdio.h>void func (int a){printf("in func a = %d\n",a);}void (*pfunc)(int); //指向函数的指针,返回voidvoid (*my_signal(int sig, void (* sighandler_t)(int h)))(int c)//void (* sighandler_t)(int h)指向函数的指针{printf("in my_signal\n"); //void (*my_signal())(int c) 这也是指向函数的指针return sighandler_t; //所以my_signal()就是一个指针}int main(){int a;printf("before invoke'void func(int)'\n");func(1); //4pfunc = func;printf("before invoke'void func(int)' by function pointer\n");pfunc(2);printf("before invoke 'my_signal(int sig,void (*sighandler_t)(int h)))'\n");pfunc = my_signal(2,func);//my_signal()就是一个指针//void (*sighandler_t)(int h)是一个指向函数的指针//handler是函数func的地址//在my_signal里,它返回sighandler_t(即func函数的地址)//func地址->pfuncprintf("before invoke 'the function pointer return by my_signal'\n"); pfunc(4); printf("before invoke 'my_signal(int sig,void (*sighandler_t)(int h)))(int c)'\n"); my_signal(5,func)(6); return 0;}
最后,在LinuxKernel中signal函数绑定信号处理函数后,确实会返回一个函数指针,它返回到是以前这个信号到处理函数指针,所以如果你存储了这个指针,那么当你处理完成以后就可以恢复以前的信号绑定。 |
|