内嵌Python import时undefined symbol错误及解决


内嵌Python import时undefined symbol错误及解决

以下代码

#include  < Python.h >
#include 
< stdio.h >
#include 
< stdlib.h >
int  main( int  argc,  char *  argv[])
{
    Py_Initialize();
    PyRun_SimpleString(
"from time import time, ctime/n"
                                          
"print 'Today is', ctime(time())/n");
    Py_Finalize();

    
return 0;
}

编译命令:
g++ -w -g -o testpy testpy.C -I/usr/local/include/python2.5
-L/usr/local/lib/python2.5/config -lpython2.5 -lpthread -ldl -lutil

运行结果:
Traceback (most recent call last):
  File "", line 1, in ?
ImportError: /usr/lib/.../lib-dynload/timemodule.so: undefined symbol: PyInt_FromLong

(仅部分系统会出错)

网上搜索到的同样的错误现象:
* Why couldn't I import the time module?

A:Why couldn't I import the time module?      

简单的一句:import time
在Python解释器下执行没有问题,可一嵌入到C中(我在C程序里用PyRun_SimpleFile()执行)就会出错,WHY?

Traceback (most recent call last):
File "test.py", line 1, in ?
import time
ImportError: /usr/lib/python2.2/lib-dynload/timemodule.so: undefined symbol: PyInt_FromLong

编辑者: xiaowu (03-11-09 21:16)

还有:
* ImportError
* undefined symbol: PyInt_FromLong

可以看到对问题的描述和寻求帮助的很多,猜测原因的也有一些,可是没有一个回复是干脆地解决问题。

其实,这一问题已经在Python的帮助手册中说明,
见Extending and Embedding部份,5.6节 Linking Requirements.
不同系统Python的配置不同,可使用以下Python命令获取额外的编译选项。

>>> import distutils.sysconfig
>>> distutils.sysconfig.get_config_var('LINKFORSHARED')
'-Xlinker -export-dynamic'

加入"-Xlinker -export-dynamic"编译选项后,内嵌import终于成功了!

这一选项是让gcc将"-export-dynamic"选项送入链接程序,输出符号给动态库使用。
即输出PyInt_FromLong给timemodule.so使用。
如果没有输出PyInt_FromLong,就会报
ImportError: /.../timemodule.so: undefined symbol: PyInt_FromLong

libPython库中虽然有PyInt_FromLong这个符号,可是链接程序查到该符号没有用到,所以缺省是不链入执行程序的。
而Python import动态调用了timemodule.so,就会用到PyInt_FromLong.
所以应该告知链接程序,不要丢弃符号。

参考:
* undefined symbol: PyExc_FloatingPointError
* Python embeding
* 5.6 Linking Requirements
* 如果把一个静态库整体编进一个动态库?


 

你可能感兴趣的:(Python)