最近写了个小程序,用到了pow()函数,我们都知道需要添加头文件<math.h>,以及在gcc编译的侍侯加上-lm选项,但是这个-lm选项的位值也是有讲究的,我之前的makefile如下:

solution:solution.o
     gcc  -lm solution.o  -o solution 
solution.o:solution.c
     gcc  -c  solution.c

报出错误:undefined reference to pow

原因是-lm的位置在solution.o之前,所以编译器链接的时侯先去链接lm库,之后无法解释solution.o里的pow函数

对于C/C++编译而言,读取编译选项是按照从左到右的顺序执行的。那么当编译器遇到源文件的时候,就开始对源文件中用到的函数进行解析,找到相对应的函数的函数体或者说是实现(Definition of Function)。这个过程是按照先遇到不能解析的函数(unresolved function),然后在源文件选项后面的一些选项中寻找可能的函数体的信息,是这样的一个顺序进行的。那么我们可以发现对于 Makefile,由于包含函数体或者函数定义信息的编译选项出现在源文件之前,那么当编译器在源文件中遇到不能解析的函数时,在源文件之后的选项中寻找相关的信息,那么就出现了编译错误,也就是无法找到相关的函数定义。

所以正确的makefile如下:

solution:solution.o
     gcc  solution.o  -o solution  -lm
solution.o:solution.c
     gcc  -c  solution.c

总结:始终将-l库函数选项放在编译命令的最右边



blog comments powered by Disqus