前言
为了使用C++ 编写python的扩展程序, 需要使用pybind11, pybind11使用比较简单,文档也比较详细。下面本人分别在Ubuntu和Windows系统上测试使用pybind11
开发/测试环境
Ubuntu系统
- Ubuntu 18.04
- pybind11
- Anaconda3, with python 3.6
- cmake
Windows系统
- win10 64bit
- Microsoft Visual Studio 2017
- Anaconda3, with python 3.7
pybind11
https://github.com/pybind/pybind11
data:image/s3,"s3://crabby-images/3e2ef/3e2ef7e748792395b099df6f97f0b01cc5c025b5" alt=""
data:image/s3,"s3://crabby-images/c6eef/c6eef1a033b6fb15211d0d15ca471060338afe57" alt=""
data:image/s3,"s3://crabby-images/df3cc/df3cc3355dcf13b2a5ebf26b02f65660ef244854" alt=""
data:image/s3,"s3://crabby-images/9dd7b/9dd7b400a6bbcc09f8bb3a950c3cd5b3aa627e0a" alt=""
安装配置
下载pybind11
git clone https://github.com/pybind/pybind11.git
安装pytest
pip install pytest
编译安装
mkdir build
cd build
cmake ..
cmake --build . --config Release --target check
data:image/s3,"s3://crabby-images/04497/044976b957a286eaa7d99223c3a4f1eb97961d05" alt=""
data:image/s3,"s3://crabby-images/0bf4d/0bf4d169f39eb29f5703d903b569607e28f73e52" alt=""
编译好的动态库
data:image/s3,"s3://crabby-images/6fa6e/6fa6e74a57bf65233f095e12db9ce5eb41c420b8" alt=""
编译pybind11文档
在doc目录下
make html
data:image/s3,"s3://crabby-images/dd7f0/dd7f0c5a548515b48b0149123b18eb33169a1d5a" alt=""
Ctrl + H
显示隐藏文件
data:image/s3,"s3://crabby-images/611ca/611ca3c2b89558f71e1d02ad64ab999c7c9956b0" alt=""
data:image/s3,"s3://crabby-images/0796d/0796db4e84708209973df66db8b6392de5b387b3" alt=""
data:image/s3,"s3://crabby-images/90a0e/90a0ea4345d6a0f2439547af4371f830c2b9acbb" alt=""
Windows系统
Requires
- Microsoft Visaul Studio 2017 x64
- Anaconda3 , with python 3.7
Pybind11配置安装
首先直接下载pybind11
https://github.com/pybind/pybind11
data:image/s3,"s3://crabby-images/421ad/421adc22eed82edad5e446f6df67a2b0ac1d4541" alt=""
pybind11是 header-only的,因此不需要编译动态链接库,直接使用即可。
使用VS2017测试
使用C++编写python扩展(python调用C++)
新建一个vs c++工程
data:image/s3,"s3://crabby-images/91632/91632862951d6570495b1945b08b050202148857" alt=""
工程配置:
- 设置编译输出类型
- 添加include包含
- 添加lib路径
- 链接器添加lib
1. 编译输出类型
data:image/s3,"s3://crabby-images/b593b/b593b60f4ea32ce3174241da76f8d40bac7d27ed" alt=""
data:image/s3,"s3://crabby-images/2a022/2a022c8fca6f04327ac72888cfcff04f885ee00b" alt=""
2. include包含路径
- python/include
-
pybind11/include
image.png
data:image/s3,"s3://crabby-images/47fa7/47fa769c476857b01d6754f5f1d01170b4139bb3" alt=""
data:image/s3,"s3://crabby-images/4c901/4c901ed9d1a94932f0c5f9c30be919370f9057f6" alt=""
3. lib路径
data:image/s3,"s3://crabby-images/2825e/2825ed502d82e234a3ea826e1566c77d5da137eb" alt=""
data:image/s3,"s3://crabby-images/52e24/52e2450c998a88af5e25ed82e2a1c014928e7cf7" alt=""
4. 链接器配置
data:image/s3,"s3://crabby-images/f3bb0/f3bb0120cac421f91008e4a51bc94bf0f1671d57" alt=""
data:image/s3,"s3://crabby-images/77255/7725520e21ad0e2121547fda580726b3715e97b1" alt=""
C++代码
很简单的一个代码,编写一个python扩展模块,模块中包含一个函数foo()
这是一个lamda
函数。
#include<pybind11/pybind11.h>
namespace py = pybind11;
PYBIND11_MODULE(example, m) {
m.doc() = "pybind11 example module";
// Add bindings here
m.def("foo", []() {
return "Hello, World!";
});
}
编译生成pyd, lib
(改名为example)
data:image/s3,"s3://crabby-images/3e892/3e8925cf5b9933755994d7c9e9ad6482c7bc8854" alt=""
测试pyd扩展
首先,打开至x64/Release目录, 打开power shell窗口:
data:image/s3,"s3://crabby-images/c7972/c7972babd2f97606ad182bc8bd8020320c906dc5" alt=""
然后,输入python
打开python解释器:
data:image/s3,"s3://crabby-images/cfe88/cfe882261b943712149da54aab7388a0112ec35f" alt=""
输入python代码:
import example
print(example.foo())
data:image/s3,"s3://crabby-images/0c210/0c21061b34b3349e5decb99e0c2ebaa6672a0004" alt=""
继续写一个复杂的例子
计算加,减,乘,除
#include<pybind11/pybind11.h>
namespace py = pybind11;
PYBIND11_MODULE(example, m) {
m.doc() = "pybind11 example module";
// Add bindings here
m.def("foo", []() {
return "Hello, World!";
});
m.def("foo2", []() {
return "This is foo2!\n";
});
m.def("add", [](int a, int b) {
return a + b;
});
m.def("sub", [](int a, int b) {
return a - b;
});
m.def("mul", [](int a, int b) {
return a * b;
});
m.def("div", [](int a, int b) {
return static_cast<float>(a) / b;
});
}
data:image/s3,"s3://crabby-images/3989c/3989c43d6d33d765d3da99fd3a9e132273ab70dc" alt=""
示例2:
data:image/s3,"s3://crabby-images/006c1/006c197d5d4d2bfee40c878e7a64a6be3c942f0d" alt=""
代码
#include<iostream>
#include<pybind11/pybind11.h>
namespace py = pybind11;
/*
file:///D:/pybind11-master/docs/.build/html/basics.html
*/
# if 1
int add(int a, int b) {
return a + b;
}
int add2(int a, int b) {
return a + b;
}
int add3(int a, int b) {
return a + b;
}
PYBIND11_MODULE(demo2, m) {
m.doc() = "example module";
//函数名称, 函数指针, 描述
m.def("add", &add, "A function which adds two numbers");
// keyword arguments
//py::arg("a")
m.def("add2", &add2, "A function which adds two numbers", py::arg("a"), py::arg("b"));
//default arguments
m.def("add3", &add3, "A function which adds two numbers", py::arg("a") = 10, py::arg("b") = 5);
//Exporting variables
m.attr("num1") = 100;
py::object world = py::cast("World");
m.attr("what") = world;
}
#endif
data:image/s3,"s3://crabby-images/eacc1/eacc112e2659562e08d0ae09a0faa7b4b31632bb" alt=""
网友评论