下面我寫了一個hello world程式,一起看看吧:
// filename: main.c #include <stdio.h> int main(void) { printf("hello wolrd!\n"); return(-); }
編譯執行:gcc main.c && ./a.out
現在我們看看在當前shell中返回上一個執行過程的傳回值是多少,是 “-1” 嗎?
inuyasha@inuyasha-Aspire-4741:~/案頭$ gcc main.c && ./a.out
hello world!
inuyasha@inuyasha-Aspire-4741:~/案頭$ echo $?
255
啊,結果為什麼 "255"呢?調用一個程式, 程式退出-1, 得到結果不是“-1”嗎?
以下引用自:http://www.laruence.com/2012/02/01/2503.html
這個問題簡單的說, 是因為exit或者main函數中的return, 只能使用0~255之間的值. -1 的unsigned值就是255.
那麼複雜點的說呢?
我們知道, 在Shell中, 運行一個命令, 一個程式, 都是fork一個子進程(然後exec)來執行的, 而這個程式的退出碼, 被Shell(父進程), 通過wait來收集而後報告給我們的.
pid_twait(int *statloc);
而對於wait來說, 曆史上原因, 他將通過statloc返回一個16bit的interge(現在也有用32位表示的, 但是會相容已有的設計). 這16bits的interge中, 高8位就是程式退出的值(exit, 或者return), 而低八位表示導致這個程式退出的訊號(其中一位表示是否有Core檔案產生), 如果程式是正常退出, 那麼低八位為0[1].
所以, 如果我們返回-1, 並且因為我們是正常退出, 所以Shell通過wait收集到的子進程退出狀態是:
11111111 00000000
而高八位作為unsigned, 就是255了.
另外, 補充一下, 在Linux的內建Shell命令中, 很多都會遵守一個退出狀態代碼的約定, 具體的值對應的意思[2]:
Exit Code Number |
Meaning |
Example |
Comments |
1 |
Catchall for general errors |
let “var1 = 1/0″ |
Miscellaneous errors, such as ”divide by zero” and other impermissible operations |
2 |
Misuse of shell builtins (according to Bash documentation) |
empty_function() {} |
Seldom seen, usually defaults to exit code 1 |
126 |
Command invoked cannot execute |
|
Permission problem or command is not an executable |
127 |
“command not found” |
illegal_command |
Possible problem with $PATH or a typo |
128 |
Invalid argument to exit |
exit 3.14159 |
exit takes only integer args in the range 0 – 255 (see first footnote) |
128+n |
Fatal error signal ”n” |
kill -9 $PPID of script |
$? returns 137 (128 + 9) |
130 |
Script terminated by Control-C |
|
Control-C is fatal error signal 2, (130 = 128 + 2, see above) |
255* |
Exit status out of range |
exit -1 |
exit takes only integer args in the range 0 – 255 |