POSIX兼容的平台上,SIGFPE是當一個進程執行了一個錯誤的算術操作時發送給它的信號。SIGFPE的符號常量頭文件signal.h中定義。因為在不同平台上,信號數字可能變化,因此常使用信號名稱。[1]

SIGFPE
描述錯誤的算術操作
默認動作進程異常終止
SA_SIGINFO宏
FPE_INTDIV 整數除以零
FPE_INTOVF 整數上溢
FPE_FLTDIV 浮點除以零
FPE_FLTOVF 浮點上溢
FPE_FLTUND 浮點下溢
FPE_FLTRES 浮點結果不准
FPE_FLTINV 無效浮點操作
FPE_FLTSUB 浮點下標越界

語源

SIG是信號名的通用前綴FPEfloating-point exception(浮點異常)的首字母縮略字。產生SIGFPE信號時並非一定要涉及浮點算術,之所以不修改名字是因為這麼做會破壞向下兼容性

描述

導致SIGFPE被發送給進程的原因有很多。一個常見的例子是由於一個意外輸入導致的溢出,或者在程序構造中的錯誤。

SIGFPE可以被處理。也就是說,程序員可以指定他們在接收到信號時想要的動作,例如調用一個子程序,忽略事件等。

在特定情形下,忽略SIGFPE可能導致程序出現意料之外的行為,包括但不限於由於不斷重試違規操作而導致程序掛起。但是,忽略並非由計算造成的SIGFPE信號是安全的,例如通過kill系統調用發送的那些。

一個通常的疏忽是認為除以零是SIGFPE的唯一來源。在一些架構上(包括IA-32[來源請求]),使用INT_MIN(最小的可以被表示的負整數值)除以-1的整數除法也會觸發這個信號,因為商是一個無法被表示的正數。(比如8位有符號整數可以表示-128、+127和它們之間的整數。-128÷-1=+128 > +127,因此無法被表示而產生溢出並觸發此信號)

例子

這是一個嘗試執行一個稱為整數除以零,或FPE_INTDIV的錯誤算術運算的ANSI C程序的例子。

int main()
{
       int x = 42/0;
       return 0; /* Never reached */
}

在一個運行LinuxIA-32上編譯運行,產生下列內容:

$ gcc -o sigfpe sigfpe.c
sigfpe.c: In function ‘main’:
sigfpe.c:3: warning: division by zero
$ ./sigfpe
Floating point exception (core dumped)

一個來自gdb棧跟蹤顯示在main函數中發生了SIGFPE信號:

Program received signal SIGFPE, Arithmetic exception.
0x08048373 in main ()

參考

參閱