Python调用C++,通过Pybind11制作Python接口


Posted in Python onOctober 16, 2018

我是在ubuntu系统进行实验的,所以和window可能会有区别。

python调用C/C++有不少的方法,如boost.python, swig, ctypes, pybind11等,这些方法有繁有简,而pybind11的优点是对C++ 11支持很好,API比较简单,现在我们就简单记下Pybind11的入门操作。

1. pybind11简介与环境安装

pybind11是一个轻量级的只包含头文件的库,它主要是用来在已有的 C++代码的基础上做扩展,它的语法和目标非常像Boost.Python,但Boost.Python为了兼容现有的基本所有的C++编译器而变得非常复杂和庞大,而因此付出的代价是很多晦涩的模板技巧以及很多不必要的对旧版编译器的支持。Pybind11摒弃了这些支持,它只支持python2.7以上以及C++ 11以上的编译器,使得它比Boost.Python更加简洁高效。

为了使用pybind11,我们需要支持C++ 11标准的编译器(GCC 4.8以上,VS 2015 Update 3以上)以及python 2.7以上的版本,还需要下载CMake,有了这些以后,

cmake教程可以参考://3water.com/article/148903.htm

  1. 首先,我们从 pybind11 github网址:https://github.com/pybind/pybind11 上下载源码。
  2. cmake工程之前,要先安装pytest pip install pytest,否则会出错
  3. 用CMake编译并运行测试用例:
进入pybind11的目录,
cd tests
cmake ..
cmake --build . --config Release --target check

如果所有测试用例都通过了,说明安装成功了。

2. python调用C++

下载编译好pybind11之后,我们就可以开始对着官方的pybind11 Tutorial进行学习了,详细的入门教程及语法请参考官方文档,这里,我们简单演示下如何编写供python调用的C++模块.
首先,我们编写一个C++源文件,命名为example.cpp

#include <pybind11/pybind11.h>
namespace py = pybind11;
int add(int i, int j)
{
 return i + j;
}
PYBIND11_MODULE(example, m)
{
 // optional module docstring
 m.doc() = "pybind11 example plugin";
 // expose add function, and add keyword arguments and default arguments
 m.def("add", &add, "A function which adds two numbers", py::arg("i")=1, py::arg("j")=2);
 // exporting variables
 m.attr("the_answer") = 42;
 py::object world = py::cast("World");
 m.attr("what") = world;
}

2.1 使用window编译

我没有实验,所以可以参考别的教程

2.2 CMake的编译方法

当然,我们也可以使用CMake进行编译。首先写一个CMakeLists.txt

cmake_minimum_required(VERSION 2.8.12)
project(example) 
add_subdirectory(pybind11)
pybind11_add_module(example example.cpp)

这里要求example.cpp放在和pybind11同一级的目录下,因为我们在CMakeLists.txt中调用了同目录pybind11和同目录的example.cpp文件。在当前目录下执行

cmake .
make

会生成example.cpython-36m-x86_64-linux-gnu.so文件。这个文件就是python可以调用的文件。还是在相同目录下运行python,进入python命令行

import example
example.add(3, 4)
[out]: 7

3、中级调用

上面是一个简单的例子,有时我们需要的功能可能很复杂。

生成模型设计库调用问题。

比如你的cpp文件中引用了其他的第三方库,这个时候我们生成的so文件,可能是需要依赖第三方库的。

本地myopencv.cpp文件

#include <pybind11/pybind11.h>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/imgproc.hpp>
#include <string>
namespace py = pybind11;
void read_img(std::string img_path)
{
 cv::Mat image = cv::imread(img_path, CV_LOAD_IMAGE_COLOR);
}
PYBIND11_MODULE(myopencv, m)
{
 m.def("read_img", &read_img, "get image size");
}

CMakeLists.txt可以写成下面这样

cmake_minimum_required(VERSION 2.8.12)
project(myopencv)
add_subdirectory(pybind11)
pybind11_add_module(myopencv myopencv.cpp)

通过cmkae编译通过

cmake .
make

产生myopencv.cpython-36m-x86_64-linux-gnu.so文件
在python中调用

import myopencv

目前这里发现有问题,还没有调通

数据类型不匹配的问题

比如opencv在python中是numpy的类型,但是在c++中cv.Mat的类型

对于opencv的类型不匹配问题,github上已经给出了解决方法,

cpp和h文件下载地址 https://github.com/edmBernard/pybind11_opencv_numpy

如何你需要什么可以去github上搜索,或者自己去实现。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。如果你想了解更多相关内容请查看下面相关链接

Python 相关文章推荐
MySQLdb ImportError: libmysqlclient.so.18解决方法
Aug 21 Python
python实现多线程的两种方式
May 22 Python
Python实现一个Git日志统计分析的小工具
Dec 14 Python
对numpy Array [: ,] 的取值方法详解
Jul 02 Python
flask中的wtforms使用方法
Jul 21 Python
Python编程图形库之Pillow使用方法讲解
Dec 28 Python
详解Python数据分析--Pandas知识点
Mar 23 Python
python机器学习库scikit-learn:SVR的基本应用
Jun 26 Python
python做接口测试的必要性
Nov 20 Python
python对Excel按条件进行内容补充(推荐)
Nov 24 Python
python3实现飞机大战
Nov 29 Python
对Keras自带Loss Function的深入研究
May 25 Python
Python之inspect模块实现获取加载模块路径的方法
Oct 16 #Python
django 将model转换为字典的方法示例
Oct 16 #Python
python复制列表时[:]和[::]之间有什么区别
Oct 16 #Python
使用Python编写Prometheus监控的方法
Oct 15 #Python
python取数作为临时极大值(极小值)的方法
Oct 15 #Python
Python文件监听工具pyinotify与watchdog实例
Oct 15 #Python
Python并行分布式框架Celery详解
Oct 15 #Python
You might like
PHP 高手之路(二)
2006/10/09 PHP
php中通过curl smtp发送邮件
2012/06/05 PHP
关于file_get_contents返回为空或函数不可用的解决方案
2013/06/24 PHP
php判断GIF图片是否为动画的方法
2020/09/04 PHP
基于node.js的快速开发透明代理
2010/12/25 Javascript
GRID拖拽行的实例代码
2013/07/18 Javascript
jQuery避免$符和其他JS库冲突的方法对比
2014/02/20 Javascript
Javascript遍历Html Table示例(包括内容和属性值)
2014/07/08 Javascript
鼠标悬浮显示二级菜单效果的jquery实现
2014/10/29 Javascript
JS实现FLASH幻灯片图片切换效果的方法
2015/03/04 Javascript
javascript实现客户端兼容各浏览器创建csv并下载的方法
2015/03/23 Javascript
artDialog+plupload实现多文件上传
2016/07/19 Javascript
JS实现屏蔽网页右键复制及ctrl+c复制的方法【2种方法】
2016/09/04 Javascript
Javascript blur与click冲突解决办法
2017/01/09 Javascript
简单的jQuery拖拽排序效果的实现(增强动态)
2017/02/09 Javascript
js+html5实现页面可刷新的倒计时效果
2017/07/15 Javascript
微信小程序6位或多位验证码密码输入框功能的实现代码
2018/05/29 Javascript
vue中设置height:100%无效的问题及解决方法
2018/07/27 Javascript
JS实现的input选择图片本地预览功能示例
2018/08/29 Javascript
微信小程序API—获取定位的详解
2019/04/30 Javascript
nodejs的安装使用与npm的介绍
2019/09/11 NodeJs
使用VUE实现在table中文字信息超过5个隐藏鼠标移到时弹窗显示全部
2019/09/16 Javascript
[02:42]2014DOTA2国际邀请赛 三冰专访:我会打到Ti20
2014/07/13 DOTA
django加载本地html的方法
2018/05/27 Python
使用django实现一个代码发布系统
2019/07/18 Python
Python3实现发送邮件和发送短信验证码功能
2020/01/07 Python
Python Tkinter图形工具使用方法及实例解析
2020/06/15 Python
使用tensorflow进行音乐类型的分类
2020/08/14 Python
css3实现input输入框颜色渐变发光效果代码
2014/04/02 HTML / CSS
Gap英国官网:Gap UK
2018/07/18 全球购物
sealed修饰符是干什么的
2012/10/23 面试题
普通党员自我剖析材料
2014/10/07 职场文书
个人查摆剖析材料
2014/10/16 职场文书
医院领导班子四风问题对照检查材料
2014/10/26 职场文书
2015年个人审计工作总结
2015/04/07 职场文书
MySQL 发生同步延迟时Seconds_Behind_Master还为0的原因
2021/06/21 MySQL