当说到signal的功能时,我们都知道它会捕捉我们所指定的信号,然后调用我们所指定的信号处理函数。但它是如何捕捉我们指定的信号的呢?下面我就以msdn上关于signal的example为例,说明signal是如何捕捉信号的。

         程序如下:

[cpp] view plain copy

// crt_signal.c  

// compile with: /c  

// Use signal to attach a signal handler to the abort routine  

#include <stdio.h>  

#include <stdlib.h>  

#include <signal.h>  

#include <tchar.h>  

  

void SignalHandler(int signal)  

{  

    printf("Application aborting...\n");  

}  

  

int main()  

{  

    typedef void (*SignalHandlerPointer)(int);  

  

    SignalHandlerPointer previousHandler;  

    previousHandler = signal(SIGABRT, SignalHandler);  

      

    abort();  

}  

        先说一下signal函数的声明:void (*signal(int sig,void (*func)(int)))(int),它的返回值类型是函数指针,这个函数指针指向一个返回值为void类型,接受一个int参数的函数。实际上signal函数返回的是该函数调用前指定信号的处理函数的指针。

        回到程序中来,这个程序很简单,首先定义了一个信号处理函数SignalHandler,然后调用signal用SignalHandler处理所产生的中止信号(SIGABRT)。执行程序时,首先调用signal函数,signal函数的定义(winsig.c中)部分如下:

[cpp] view plain copy

_PHNDLR __cdecl signal(int signum,_PHNDLR sigact)  

{  

    //...  

    switch (signum) {  

            case SIGINT:  

                    //...  

                    break;  

            case SIGBREAK:  

                    //...  

                    break;  

            case SIGABRT:  

            case SIGABRT_COMPAT:  

                    oldsigact = (_PHNDLR) DecodePointer(abort_action);  

                    if(sigact!=SIG_GET)  

                    {  

                        abort_action = (_PHNDLR) EncodePointer(sigact);  

                    }  

                    break;  

            case SIGTERM:  

                    //...  

                    break;  

                }  

     //...  

}  

        我们可以看到,指定了SIGABRT,signal函数就会执行case SIGABRT下面的语句,将指向函数调用前的SIGABRT处理函数的指针赋给oldsigact,将新的处理函数编码后赋给abort_action,这一步非常重要,因为下面的abort()函数就是根据它来得到信号处理函数的。

        接下来执行abort函数,该函数会产生SIGABRT信号,其定义(abort.c中)如下:

[cpp] view plain copy

void __cdecl abort (  

        void  

        )  

{  

    _PHNDLR sigabrt_act = SIG_DFL;  

  

    //...  

  

    sigabrt_act = __get_sigabrt();  

    if (sigabrt_act != SIG_DFL)  

    {  

        raise(SIGABRT);  

    }  

    //...  

    _exit(3);  

}  

         该函数调用__get_sigabrt()取得信号处理函数sigabrt_act,然后调用raise(SIGABRT),在这个函数中调用信号处理函数。raise()的定义()如下:

[cpp] view plain copy

int __cdecl raise (  

        int signum  

        )  

{  

        _PHNDLR sigact;  

        _PHNDLR *psigact;  

        switch (signum) {  

  

                case SIGINT:  

                        sigact = *(psigact = &ctrlc_action);  

                        siglock++;  

                        break;  

  

                case SIGBREAK:  

                        sigact = *(psigact = &ctrlbreak_action);  

                        siglock++;  

                        break;  

  

                case SIGABRT:  

                case SIGABRT_COMPAT:  

                        sigact = *(psigact = &abort_action);  

                        siglock++;  

                        break;  

  

                case SIGTERM:  

                        sigact = *(psigact = &term_action);  

                        siglock++;  

                        break;  

  

                case SIGFPE:  

                case SIGILL:  

                case SIGSEGV:  

                        ptd = _getptd_noexit();  

                        if (!ptd)  

                            return (-1);  

                        sigact = *(psigact = &(siglookup( signum,  

                            ptd->_pxcptacttab )->XcptAction));  

                        goto decode_done;  

                        break;  

  

                default:  

                        /* 

                         * unsupported signal, return an error 

                         */  

                        _VALIDATE_RETURN(("Invalid signal or error", 0), EINVAL, -1);  

        }  

        sigact = (_PHNDLR) DecodePointer(sigact);  

  

decode_done:  

        /* 

         * If the current action is SIG_IGN, just return 

         */  

        if ( sigact == SIG_IGN )  

                return(0);  

  

        /* 

         * If the current action is SIG_DFL, take the default action 

         */  

        if ( sigact == SIG_DFL ) {  

                _exit(3);  

        }  

  

        //...  

                (*sigact)(signum);  

        return(0);  

}  

        在case SIGABRT中将abort_action的值即信号处理函数指针赋给sigact,在最后调用函数(*sigact)(signum)完成对SIGABRT信号的处理。程序对其他信号的捕捉过程也差不多是这样。

        在我这里说的只是一个大概的流程,其中还有很多代码的细节没有涉及到,比如说对预定义的信号处理函数的判断、异常的处理等,这些还有待深入理解和研究。


【VS开发】程序如何捕捉signal函数参数中指定的信号的更多相关文章

  1. Python函数参数中的冒号与箭头

    在一些Python的工程项目中,我们会看到函数参数中会有冒号,有的函数后面会跟着一个箭头,你可能会疑惑,这些都是什么东西? 其实函数参数中的冒号是参数的类型建议符,告诉程序员希望传入的实参的类型.函数 ...

  2. python函数参数中带有默认参数list的坑

    在python中函数参数中如果带有默认参数list遇到问题 先看一段代码 def f(x,l=[]): for i in range(x): l.append(i*i) print(l) print( ...

  3. (转)python中函数参数中如果带有默认参数list的特殊情况

    在python中函数参数中如果带有默认参数list遇到问题 先看一段代码 1 2 3 4 5 6 7 8 9 def f(x,l=[]):     for i in range(x):         ...

  4. 关于cmp函数参数中的&符号

    关于cmp函数参数中的&符号 关于sort函数中的cmp函数有着不同的写法,以刚刚的整形元素比较为例 还有人是这么写的: bool cmp(const int &a, const in ...

  5. 错误“Sources”参数中指定了多次。“Sources”参数不支持重复项。

    在“Sources”参数中指定了项“”多次.“Sources”参数不支持重复项. Asp.Net关于错误“Sources”参数中指定了多次.“Sources”参数不支持重复项. “Sources”参数 ...

  6. PowerShell控制台输出符号+函数参数类型指定+文本内容读取

    There are several ways: Write-Host: Write directly to the console, not included in function/cmdlet o ...

  7. Linux下利用signal函数处理ctrl+c等信号

    前言 linux下能够通过信号机制来实现程序的软中断,是一个很实用的编程方法. 我们平时在程序执行的时候按下ctrl-c.ctrl-z或者kill一个进程的时候事实上都等效于向这个进程发送了一个特定信 ...

  8. Delphi 中 函数参数中的 const 修饰符的本质以及注意事项

    来自:http://blog.csdn.net/farrellcn/article/details/9096787 ------------------------------------------ ...

  9. 通过java程序调用ant build.xml配置文件中指定的target

    一.概述 通过ant实现项目的自动化部署,jar包生成,替换,tomcat关停.启动,查看项目日志: 通过java程序调用已编辑好的ant脚本build.xml配置文件中指定的target: 文中文件 ...

随机推荐

  1. PPP

    名称 chat–调制解调器的自动对话脚本 命令格式 chat [options] script 描述 Chat程序定义了一个计算机和调制解调器之间对话交流,其主要目的是用来在本地PPPD和远端PPPD ...

  2. P4555 【[国家集训队]最长双回文串】

    不知道有没有人跟我一样数据结构学傻了 首先这道题是要求回文串,那么我们可以想到manacher算法 但由于\(manacher\)不能求出双回文子串,我们要考虑一些性质 首先对于一个回文串,删掉两边的 ...

  3. Day11:Flex布局

    参考: 来源:http://www.ruanyifeng.com/blog/2015/07/flex-grammar.html 网页布局是css的一个重点. 盒子模型 display属性 positi ...

  4. 《挑战30天C++入门极限》C/C++中字符指针数组及指向指针的指针的含义

        C/C++中字符指针数组及指向指针的指针的含义 就指向指针的指针,很早以前在说指针的时候说过,但后来发现很多人还是比较难以理解,这一次我们再次仔细说一说指向指针的指针. 先看下面的代码,注意看 ...

  5. driud 异常

    异常如下: 十二月 25, 2017 11:37:14 下午 org.apache.tomcat.util.digester.SetPropertiesRule begin警告: [SetProper ...

  6. 菜鸟的算法入门:java的链表操作

    从C语言的指针开始,我的算法之路就结束了! 今天为了找个好的实习,不得不捡起来,写了三年的web,算法落下了太多了 今天在leetcode上刷题,难在了一个简单的链表上,因此记录一下 题目:给定两个非 ...

  7. 小程序中嵌套的h5页面设置分享转发

    场景描述:当在小程序中打开h5页面时,希望小程序的转发出去的标题,图片,跳转link可以通过h5通信实现自定义. 实现方式:通过h5给小程序通信,发送标题,图片,跳转link等信息,让小程序设置分享. ...

  8. mysql使用replace和on duplicate key update区别

    实际业务使用中,有时候会遇到插入数据库,但是如果某个属性(比如:主键)存在,就做更新.通常有两种方式:1.replace into  2.on duplicate key update 但是在使用过程 ...

  9. "笨方法"学习CNN图像识别(二)—— tfrecord格式高效读取数据

    原文地址:https://finthon.com/learn-cnn-two-tfrecord-read-data/-- 全文阅读5分钟 -- 在本文中,你将学习到以下内容: 将图片数据制作成tfre ...

  10. 基于栈的指令集与基于寄存器的指令集的区别,JVM指令集实例

    现代JVM在执行Java代码的时候,通常都会将解释执行与编译执行两者结合起来 所谓解释执行,就是通过解释器来读取字节码,遇到相应的指令就去执行该指令. 所谓编译执行,就是通过即时编译器(Just In ...