2009-11-10 47 views
6

我創建一個小程序,如下所示:SQRT導致連接錯誤「未定義參照SQRT」只有當參數不是恆定

#include <math.h> 
    #include <stdio.h> 
    #include <unistd.h> 

    int main(int argc, char *argv[]) { 
    int i; 
    double tmp; 
    double xx; 

    for(i = 1; i <= 30; i++) { 
     xx = (double) i + 0.01; 

     tmp = sqrt(xx); 
     printf("the square root of %0.4f is %0.4f\n", xx,tmp); 
     sleep(1); 
     xx = 0; 
    } 

    return 0; 
} 

當我嘗試用下面的編譯此命令,我得到一個編譯器錯誤。

gcc -Wall calc.c -o calc 

回報:

/tmp/ccavWTUB.o: In function `main': 
calc.c:(.text+0x4f): undefined reference to `sqrt' 
collect2: ld returned 1 exit status 

如果我替換變量在調用的sqrt(XX)與像開方(10.2)常數,它編譯就好了。或者,如果我明確地鏈接如下:

gcc -Wall -lm calc.c -o calc 

它也工作得很好。誰能告訴我這是什麼原因?我很長一段時間一直是C程序員(我用math.h寫了類似的小程序),我從來沒有見過類似的東西。

我的gcc版本如下:

$ gcc --version 
gcc (Ubuntu 4.3.3-5ubuntu4) 4.3.3 
Copyright (C) 2008 Free Software Foundation, Inc. 
This is free software; see the source for copying conditions. There is NO 
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. 
$ 
+0

如果指定'--std = c99',它仍然需要'-lm'嗎? – 2009-11-10 23:15:28

+0

是的。奇怪的。我也嘗試使用-ansi來達到同樣的效果。 No-go without -lm – alesplin 2009-11-10 23:22:18

+4

這是gcc的一個有趣功能,您需要明確鏈接數學庫 - 它首次吸引了所有人。 編譯器可以靜靜地替換sqrt(10.2),只是增加了樂趣! – 2009-11-10 23:25:20

回答

16

如果你看一下在您使用sqrt(10.2)的情況下,編譯器的輸出,我敢打賭,你看到到sqrt()呼叫未曾作過。

發生這種情況是因爲GCC認識到它可以專門處理的幾項功能。這使其有能力做一些優化,在這種情況下,Constant folding。這種特殊功能稱爲Built-ins

如果它必須鏈接到數學庫(因爲您使用變量調用它),則需要明確鏈接它。有些操作系統/編譯器會爲你做,這就是你以前可能沒有注意到的原因。

+2

+1許多系統自動與你的數學庫鏈接。有些不。你的不是。 – 2009-11-10 23:20:32

+11

我想他想說的是,如果你使用常量的sqrt,結果也是一個常量,所以編譯器會在編譯時爲你做,而不會在代碼中調用sqrt,在鏈接時需要鏈接到libm(-lm)。 – 2009-11-10 23:20:35

+0

南方酒店管理有限。 – 2009-11-10 23:26:41