我正在尝试编译simple_xy_wr.f90 – 一个netCDF示例程序 – 在Ubuntu上使用gfortran,我必须做一些非常愚蠢的事情;我没有太多编译Fortran的经验.
首先,我已经安装了libnetcdf-dev软件包,其中包括类似的文件
/usr/lib/libnetcdf.a /usr/lib/libnetcdff.a /usr/include/netcdf.mod
所以,我试图用(各种命令)编译代码
f95 -o xy -I/usr/include/ -L/usr/lib/ -lnetcdff -lnetcdf simple_xy_wr.f90
我得到以下输出
/tmp/ccE6g7sr.o: In function `check.1847': simple_xy_wr.f90:(.text+0x72): undefined reference to `__netcdf_MOD_nf90_strerror' /tmp/ccE6g7sr.o: In function `MAIN__': simple_xy_wr.f90:(.text+0x284): undefined reference to `__netcdf_MOD_nf90_create' simple_xy_wr.f90:(.text+0x2b6): undefined reference to `__netcdf_MOD_nf90_def_dim' simple_xy_wr.f90:(.text+0x2e8): undefined reference to `__netcdf_MOD_nf90_def_dim' simple_xy_wr.f90:(.text+0x432): undefined reference to `__netcdf_MOD_nf90_def_var_manydims' simple_xy_wr.f90:(.text+0x468): undefined reference to `__netcdf_MOD_nf90_enddef' simple_xy_wr.f90:(.text+0x4aa): undefined reference to `__netcdf_MOD_nf90_put_var_2d_fourbyteint' simple_xy_wr.f90:(.text+0x4cb): undefined reference to `__netcdf_MOD_nf90_close' collect2: error: ld returned 1 exit status
我认为我包括了正确的库.例如.看来__netcdf_MOD_nf90_strerror应该在那里:
$nm /usr/lib/libnetcdff.a | grep __netcdf_MOD_nf90_strerror 000000000004a100 T __netcdf_MOD_nf90_strerror
我究竟做错了什么?
(FWIW,我看过的一些相关参考文献如下.
> undefined reference using netcdf library
> Compiling problems with gfortran and NETCDF
> Compiling and Running Fortran Programs – a basic guide
)
这意味着如果您的代码引用__netcdf_MOD_nf90_strerror,则包含此符号定义的存档(libnetcdff.a)必须出现在程序中的目标文件列表之后. libnetcdff.a本身引用了C库libnetcdf.a中的符号,因此必须在libnetcdff.a之后进行链接.所以正确的链接顺序是:
/tmp/ccE6g7sr.o libnetcdff.a libnetcdf.a
其中/tmp/ccE6g7sr.o是汇编程序从编译的源文件生成的临时目标文件.然后,编译代码的正确命令行是:
f95 -o xy -I/usr/include/ simple_xy_wr.f90 -lnetcdff -lnetcdf
在这种情况下,不直接调用链接器,而是编译器执行它. GCC编译器以相同的顺序将所有与链接相关的东西传递给名为collect2的中间实用程序,然后调用实际的链接器ld.
请注意,如果netCDF库归档的共享对象版本也存在(即有libnetcdff.so和libnetcdf.so),则链接器会更喜欢静态归档(除非使用-static选项启用静态链接)并且最终的链接阶段将被处理到运行时链接编辑器(RTLD)(Ubuntu上的/lib64/ld-linux-x86-64.so.2).在这种情况下,与问题中相同的命令行实际上会成功而没有链接错误,尽管两个库都位于引用它们的代码之前,因为RTLD在加载可执行文件时将解析缺少的符号引用文件.