編譯Linux程序出現“對‘pthread_create’未定義的引用”

  在學習Linux程序設計POXIS線程的時候,編譯一個段簡單的代碼,就出現下面的錯誤:

1 /tmp/cc2FQJTh.o:在函數‘main’中:
2 1-thread.c:(.text+0x29):對‘pthread_create’未定義的引用
3 1-thread.c:(.text+0x67):對‘pthread_join’未定義的引用
4 collect2: 錯誤: ld 返回 1

  可是在我的代碼中已經包含了pthread.h,這是什麼回事呢,百度一下,發現網上大部分的解決辦法是在gcc編譯命令加上選項:lpthread。

1 gcc -o pthread -lpthread pthread.c

  但我照着網上的方法試試,發現還是不行,錯誤提示與原來相同。後來才發現是我自己錯了,我是在目標代碼文件前加上-lpthread。但實際的做法應該是在代碼文件後面添加此選項。

複製代碼
1 finlay@finlay-Lenovo-G470:~/文檔/程序設計/Linux程序設計/12章$ gcc 1-thread.c -lpthread
2 #成功!
3 finlay@finlay-Lenovo-G470:~/文檔/程序設計/Linux程序設計/12章$ gcc -lpthread -Wall 1-thread.c
4 #失敗!
5 /tmp/ccE6dosW.o:在函數‘main’中:
6 1-thread.c:(.text+0x29):對‘pthread_create’未定義的引用
7 1-thread.c:(.text+0x67):對‘pthread_join’未定義的引用
8 collect2: 錯誤: ld 返回 1
複製代碼

  但是還有一個什麼有趣的現象,如下:

複製代碼
1 finlay@finlay-Lenovo-G470:~/文檔/程序設計/Linux程序設計/12章$ gcc -pthread 1-thread.c
2 #成功!
3 finlay@finlay-Lenovo-G470:~/文檔/程序設計/Linux程序設計/12章$ gcc -lpthread 1-thread.c
4 #失敗!
5 /tmp/ccbr6m7v.o:在函數‘main’中:
6 1-thread.c:(.text+0x29):對‘pthread_create’未定義的引用
7 1-thread.c:(.text+0x67):對‘pthread_join’未定義的引用
8 collect2: 錯誤: ld 返回 1
複製代碼

  那麼,-pthread與-lpthread的區別是什麼呢?我們可以在verbose模式下執行一下對應的gcc命令行看出來。下面是老式的直接加 -lpthread 鏈接選項的輸出結果:

複製代碼
$ gcc -v -c x.c
...
/usr/lib/gcc/i486-linux-gnu/4.2.4/cc1 -quiet -v x.c -quiet -dumpbase x.c
-mtune=generic -auxbase x -version -fstack-protector -fstack-protector -o /tmp/cch4ASTF.s
...
as --traditional-format -V -Qy -o x.o /tmp/cch4ASTF.s
...
$ gcc -v x.o -ox -lpthread
...
 /usr/lib/gcc/i486-linux-gnu/4.2.4/collect2 --eh-frame-hdr -m elf_i386 --hash-style=both
-dynamic-linker /lib/ld-linux.so.2 -ox
/usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crt1.o
/usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crti.o
/usr/lib/gcc/i486-linux-gnu/4.2.4/crtbegin.o
-L/opt/intel/Compiler/11.1/046/tbb/ia32/cc4.1.0_libc2.4_kernel2.6.16.21/lib/../lib
-L/usr/lib/gcc/i486-linux-gnu/4.2.4
-L/usr/lib/gcc/i486-linux-gnu/4.2.4
-L/usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib
-L/lib/../lib
-L/usr/lib/../lib
-L/opt/intel/Compiler/11.1/046/lib/ia32
-L/opt/intel/Compiler/11.1/046/tbb/ia32/cc4.1.0_libc2.4_kernel2.6.16.21/lib
-L/usr/lib/gcc/i486-linux-gnu/4.2.4/../../..
x.o -lpthread -lgcc --as-needed -lgcc_s --no-as-needed -lc -lgcc
--as-needed -lgcc_s --no-as-needed
/usr/lib/gcc/i486-linux-gnu/4.2.4/crtend.o /usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crtn.o
複製代碼

  下面是在編譯和鏈接時分別指定 -pthread 選項的輸出結果:

複製代碼
 1 $ gcc -v -pthread -c x.c
 2 ...
 3 /usr/lib/gcc/i486-linux-gnu/4.2.4/cc1 -quiet -v <strong>-D_REENTRANT</strong>
 4  x.c -quiet -dumpbase x.c
 5 -mtune=generic -auxbase x -version -fstack-protector -fstack-protector -o /tmp/cc205IQf.s
 6 ...
 7 as --traditional-format -V -Qy -o x.o /tmp/cc205IQf.s
 8 ...
 9 $ gcc -v x.o -ox -pthread
10 /usr/lib/gcc/i486-linux-gnu/4.2.4/collect2 --eh-frame-hdr -m elf_i386 --hash-style=both
11 -dynamic-linker /lib/ld-linux.so.2 -ox
12 /usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crt1.o
13 /usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crti.o
14 /usr/lib/gcc/i486-linux-gnu/4.2.4/crtbegin.o
15 -L/opt/intel/Compiler/11.1/046/tbb/ia32/cc4.1.0_libc2.4_kernel2.6.16.21/lib/../lib
16 -L/usr/lib/gcc/i486-linux-gnu/4.2.4
17 -L/usr/lib/gcc/i486-linux-gnu/4.2.4
18 -L/usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib
19 -L/lib/../lib
20 -L/usr/lib/../lib
21 -L/opt/intel/Compiler/11.1/046/lib/ia32
22 -L/opt/intel/Compiler/11.1/046/tbb/ia32/cc4.1.0_libc2.4_kernel2.6.16.21/lib
23 -L/usr/lib/gcc/i486-linux-gnu/4.2.4/../../..
24 x.o -lgcc --as-needed -lgcc_s --no-as-needed <strong>-lpthread</strong>
25  -lc -lgcc
26 --as-needed -lgcc_s --no-as-needed
27 /usr/lib/gcc/i486-linux-gnu/4.2.4/crtend.o /usr/lib/gcc/i486-linux-gnu/4.2.4/../../../../lib/crtn.o
複製代碼

  所以可見編譯選項中指定 -pthread 會附加一個宏定義 -D_REENTRANT ,該宏會導致 libc 頭文件選擇那些thread-safe的實現;鏈接選項中指定 -pthread 則同 -lpthread 一樣,只表示鏈接 POSIX thread 庫。由於 libc 用於適應 thread-safe 的宏定義可能變化,因此在編譯和鏈接時都使用 -pthread 選項而不是傳統的 -lpthread 能夠保持向後兼容,並提高命令行的一致性。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章