我正在尝试使用简单的C文件扩展python。我成功创建了自己的python模块,但我还需要C文件本身作为独立的可执行文件进行编译和运行。我可以成功编译它,但是当我尝试运行它时,我得到错误"无法执行二进制文件:执行格式错误"。
这是我的C源文件(hellomodule.c):
#include <Python.h>
void print_hello(const char* name){
printf("Hello %s!\n", name);
}
//Only used by Python
static PyObject*
say_hello(PyObject* self, PyObject* args)
{
const char* name;
if (!PyArg_ParseTuple(args, "s", &name))
return NULL;
//printf("Hello %s!\n", name);
print_hello("World");
Py_RETURN_NONE;
}
//Only used by Python
static PyMethodDef HelloMethods[] =
{
{"say_hello", say_hello, METH_VARARGS, "Greet somebody."},
{NULL, NULL, 0, NULL}
};
//Only used by Python
PyMODINIT_FUNC
inithello(void)
{
(void) Py_InitModule("hello", HelloMethods);
}
int main(){
print_hello("World");
}
我可以&#34;成功&#34;使用以下代码编译它没有任何错误或警告:
gcc -I /usr/include/python2.7 -c hellomodule.c -o hellomodule
制作&#34; hellomodule&#34;文件可执行文件,我运行它并得到错误:
-bash: ./hellomodule: cannot execute binary file: Exec format error
为什么会导致这样的错误?
答案 0 :(得分:1)
您正在尝试执行不是可执行文件的目标文件。要将代码编译为模块,您需要类似的东西
gcc -Wall -Werror -Wextra -O2 -I/usr/include/python2.7 \
-shared hellomodule.c -o hellomodule.so -lpython2.7
但要正确链接到所有内容并添加所有可能的包含目录,有一个名为python-config
的脚本,您只需要像这样调用它
gcc -Wall -Werror -Wextra -O2 `python-config --includes` \
-shared hellomodule.c -o hellomodule.so `python-config --libs`
甚至更好,该脚本也提供CFLAGS
和LDFLAGS
,最后
gcc -Wall -Werror -Wextra -O2 `python-config --cflags` \
-shared hellomodule.c -o hellomodule.so `python-config --ldflags`
然后,将生成的文件复制到/usr/lib/python2.7/site-packages
。
之后,您可以在这样的python脚本中加载模块
import hellomodule
目标文件是最终由链接器( ld
可能)使用的中间二进制文件,以生成最终的二进制文件。 python模块没有main()
函数,它必须是一个运行时可加载的共享对象,它导出一些预定义的符号,python解释器将用它们将模块加载到python脚本/程序中。
注意:要正确执行此操作而不是尝试创建一个Makefile,就像这样
CFLAGS = `python-config --cflags` -Wall -Werror # add more if needed
LDFLAGS = `python-config --ldflags` # add more if needed
OBJS = hellomodule.o # add more if needed
all: $(OBJS)
$(CC) $(CFLAGS) $(LDFLAGS) -shared -o modulename.so $(OBJS)
%.o: %.c
$(CC) -c $(CFLAGS) $<
确保使用标签而不是缩进空格,然后在 Makefile 和源文件所在的同一目录中运行make
。< / p>