本篇文章使用三种方法来编写c++的扩展
第一种方法是最基础也是最麻烦的方法, 手动编译
第二种方法是正规的方法,
第三种方法是使用boost简化第二种方法, 最为简单
几个文章
C/C++与python互相调用
如何实现 C/C++ 与 Python 的通信?
boost安装
boost配置
boost_python例子
遇到的几个问题
- vs的开发者命令提示在哪?
工具->命令行-> 开发者命令提示 | PowerShell
- 执行vs开发者命令提示 的编译命令, 报错,
库计算机类型“x64”与目标计算机类型“x86”冲突
这个原因我也没找到, 但是换一个python就解决了 - 编译成pyd之后执行报错:
importerror: dynamic module does not define init function (init module)
原因在于入口签名要定义好格式: (第一次在百度上发现有用的)
IronPython 将python源码编译成dll,用python调用报错ImportError: dynamic module does not define…
Python 3.X扩展的入口函数签名如下:
PyMODINIT_FUNC PyInit_xxxxxxx(void); // 其中xxxxxxx是module名,必须保持一致
详见:http://docs.python.org/3/extending/
Python 2.X扩展的入口函数签名如下:
PyMODINIT_FUNC initxxxxxxx(void) // 其中xxxxxxx是module名,必须保持一致
详见:http://docs.python.org/2/extending/
第一种方法:p
- 配置python环境, 见第二种方法python配置
- 写代码, 如下
- 手动编译
举个例子: 比如如下代码:
#pragma once
#include"Python.h"
#include<iostream>
static PyObject* _print_hello(PyObject* self, PyObject* args)
{
std::cout << "hello world" << std::endl;
Py_RETURN_NONE;
}
static PyMethodDef cpp_methods[] = {
{"print_hello", _print_hello, METH_VARARGS, ""},
{NULL, NULL, 0, NULL}
};
// !!!看这里, 看这里 !!!
// 这里的函数名一定要注意, 你定义Py_InitModule中的的module是什么, 这里就要写上 initxxx
PyMODINIT_FUNC initcpp_module(void)
{
(void)Py_InitModule("cpp_module", cpp_methods);
}
举例
-
就用上面的一个代码, 随便写在什么cpp文件里
然后打开 开发者命令行提示, 执行如下代码
cl.exe /LD 你的文件.cpp /o cpp_module.pyd -I C:\python27\include C:\python27\libs\python27.lib
- cpp_module.pyd主要是为了和上面代码中定义的module名称一样
- include 和 libs就是你的python文件夹下的include文件夹和libs文件夹
-
建立一个文件夹, 把pyd文件放进去
3.建立一个文件, 写入如下代码, 运行:
from cpp_module import print_hello
print_hello()
执行就会打印hello world
第二种方法
- 给项目配置python环境
- 给项目配置类型为dll (就是输出结果是dll)
- 编译
python设置:
!!!!!!!!!!!!!!!注意勾选所有配置, x64!!!!!!!!!
include
:
libs
:
附加依赖项
:
项目设置为dll
配置类型设置为dll
code
然后创建一个Test.cpp文件, copy下面一段代码, 直接生成. 具体的注意事项都在代码中注释了, 别忘了, 项目也设置成x64的
#include <Python.h>
// 具体的PyArg_ParseTuple的用法
// 参考 http://web.mit.edu/people/amliu/vrut/python/ext/parseTuple.html
PyObject* my_mul(PyObject* self, PyObject* args)
{
long a, b;
PyArg_ParseTuple(args, "ll", &a, &b);
return PyInt_FromLong(a * b);
}
PyObject* my_add(PyObject* self, PyObject* args)
{
long a, b, c;
PyArg_ParseTuple(args, "lll", &a, &b, &c);
return PyInt_FromLong(a + b + c);
}
// 具体的定义方式 ,参考:
// https://www.rddoc.com/doc/Python/3.6.0/zh/c-api/structures/#METH_VARARGS
static PyMethodDef superfastcode_methods[] = {
{ "my_mul", (PyCFunction)my_mul, METH_VARARGS, nullptr },
{ "my_add", (PyCFunction)my_add, METH_VARARGS, nullptr },
{ nullptr, nullptr, 0, nullptr }
};
/*
* file_name = Test
* module_name = file_name
* Py_InitModule(module_name, 函数定义函数)
*/
PyMODINIT_FUNC initTest(void)
{
(void)Py_InitModule("Test", superfastcode_methods);
}
点击生成, 就会在project_dir \ x64 \ release
下生成dll文件, 直接改名后缀pyd, 放到测试文件同目录下:
''# -*- coding: utf-8 -*-
from Test import my_mul, my_add
print my_add(1, 2, 3)
print my_mul(2, 4)
第三种方法
- 给项目配置python环境
- 给项目配置boost环境
- 设置项目配置为dll
boost相关的安装, 编译, 配置如下
boost安装
boost配置
boost_python例子
然后写代码:
model命名这里为hello
,如果输出的不是hello.dll
就改名为hello.dll
, 然后改名为hello.pyd
// 这个define必须要写, 用的是静态的lib
#define BOOST_PYTHON_STATIC_LIB
#include <boost/python.hpp>
char const* greet()
{
return "hello world";
}
BOOST_PYTHON_MODULE(hello)
{
using namespace boost::python;
def("greet", greet);
}
复制pyd
到目标目录下, 然后写python
import hello
print hello.greet()