Python扩展(pybind11混编)
背景介绍
pybind11是一个基于C++11标准的模版库. 与Boost.Python类似, pybind11主要着眼于创建C++代码的Python封装, 并为其提供了一套轻量级的解决方案.安装与代码示例
①. 安装C++编译器(各平台略有不同, 支持C++11标准即可)
②. 安装cmake工具(官网下载安装即可, 用于组织C++工程)
③. 安装Python解释器(官网下载安装即可)
④. 安装pybind11库
终端运行: pip3 install pybind11
⑤. 获取pybind11库相关目录
解释器内运行:import pybind11
pybind11.get_cmake_dir() # 获取cmake目录
pybind11.get_include() # 获取include目录⑥. 待封装之C++源码
本文以一个main.cpp源文件为例, 简要给出一个函数与一个类的封装示例, 代码如下,#include <string>
#include <iostream> #include <pybind11/pybind11.h>
#include <pybind11/eigen.h> int MyFunc(int i, int j)
{
return i + j;
} class MyClass
{
public:
MyClass(const std::string& msg) : msg_(msg) {} void printMsg()
{
std::cout << this->msg_ << std::endl;
} Eigen::VectorXd add(const Eigen::VectorXd& lhs, const Eigen::VectorXd& rhs)
{
Eigen::VectorXd ret = lhs + rhs;
return ret;
} std::string msg_;
}; PYBIND11_MODULE(testlib, m) // 此处设置模块名为testlib
{
m.doc() = "This is a test library"; m.def("MyFunc", &MyFunc, "my first function",
pybind11::arg("i") = 1, pybind11::arg("j") = 2); pybind11::class_<MyClass>(m, "MyClass")
.def(pybind11::init<const std::string&>())
.def("printMsg", &MyClass::printMsg)
.def("add", &MyClass::add, pybind11::arg("lhs"), pybind11::arg("rhs"))
.def_readwrite("msg_", &MyClass::msg_);
}其中, MyFunc是待导出函数, MyClass是待导出类. 注意, 上例含eigen库(C++)与numpy库(Python)之映射, 无eigen库的小伙伴可以注释相关内容.
cmake工程示例
配合上述main.cpp源文件, CMakeLists.txt文件内容如下,cmake_minimum_required(VERSION 3.15) set(CMAKE_BUILD_TYPE "Release")
set(CMAKE_CXX_STANDARD 11)
project(test_lib) set(test_srcs
main.cpp
) set(pybind11_DIR "/opt/homebrew/lib/python3.9/site-packages/pybind11/share/cmake/pybind11") # 此处设置pybind11之cmake目录, 即: pybind11.get_cmake_dir()
find_package(pybind11 REQUIRED)
pybind11_add_module(testlib ${test_srcs}) # 此处设置模块名为testlib
target_include_directories(testlib PUBLIC "/Users/xxhbdk/MyLibs/eigen-3.4.0") # 此处附加包含eigen库目录当前工程结构如下,

编译及效果展示
终端运行如下命令编译Python动态库:mkdir build # 创建编译目录
cd build
cmake ..
make运行完成后, 笔者build目录下生成了Python动态库文件"testlib.cpython-39-darwin.so". 随后即可在Python环境中使用之, 测试效果如下,

可以看到, 接口导出整体符合预期.注意事项
①. C++源文件中模块名需要与cmake工程文件中模块名保持一致;
②. 本文着重阐述pybind11配合cmake之通用流程, 具体API使用细节, 请大家参考官方文档等资料.参考文档
①. https://pybind11.readthedocs.io/en/stable/②. https://cmake.org/cmake/help/latest/- 补充1(C++调用Python)
①. 待调用之Python源码
本文以一个my_func.py源文件为例, 简要给出一个函数示例, 代码如下,
def MyFunc(i, j):
return i + j②. 待编译之C++源码
本文以一个main.cpp源文件为例, 简要给出一个C++调用Python函数之示例, 代码如下,#include <iostream>
#include <pybind11/embed.h> int main()
{
pybind11::scoped_interpreter guard; // 初始化python解释器 pybind11::module my_func = pybind11::module::import("my_func");
int i = 11;
int j = 22;
pybind11::object ret = my_func.attr("MyFunc")(i, j);
int n = ret.cast<int>();
std::cout << i << " + " << j << " = " << n << std::endl;
}③. cmake工程示例
配合上述main.cpp源文件, CMakeLists.txt文件内容如下,cmake_minimum_required(VERSION 3.15) set(CMAKE_BUILD_TYPE "Release")
set(CMAKE_CXX_STANDARD 11)
project(test_cppInvokePy) set(test_srcs
main.cpp
) add_executable(main ${test_srcs})
set(pybind11_DIR "/opt/homebrew/lib/python3.9/site-packages/pybind11/share/cmake/pybind11") # 此处设置pybind11之cmake目录, 即: pybind11.get_cmake_dir()
find_package(pybind11 REQUIRED)
target_link_libraries(main PUBLIC pybind11::embed)④. 编译及效果展示
当前工程结构如下,
终端运行如下命令编译C++可执行文件,
mkdir build # 创建编译目录
cd build
cmake ..
make运行完成后, 笔者build目录下生成了可执行文件"main". 随后将Python文件my_func.py拷贝至此build目录下.终端运行可执行文件main, 测试效果如下,

可以看到, 执行结果符合预期, Python模块调用成功.
- 补充1 - 参考文档
①. https://www.yuque.com/u461675/pcadi1/hf4fha#e82c4d4d - 补充2(Global Interpreter Lock)
当执行流从Python侧进入C++侧时, GIL总是持有的. 因此, 如果C++侧代码长时间运行, 且不释放GIL, 则Python侧多线程可能无法达到预期的运行效果(如: UI运行阻塞等).
因此, 通过Python调用C++时, 若C++侧代码执行时间较长且具备Python侧多线程需求, 建议在C++代码入口处释放GIL.
释放GIL之方法①(功能代码执行处): pybind11::gil_scoped_release release;
释放GIL之方法②(模块接口定义处): pybind11::call_guard<pybind11::gil_scoped_release>(); - 补充2 - 注意事项
①. C++侧多线程不受GIL影响. - 补充2 - 参考文档
①. https://pybind11.readthedocs.io/en/stable/advanced/misc.html#global-interpreter-lock-gil
Python扩展(pybind11混编)的更多相关文章
- python 与 matlab 混编
用于 Python 的 MATLAB 引擎 API 快速入门 安装用于 Python 的 MATLAB 引擎 API Matlab的官方文档中介绍了 Matlab 与其余编程语言之间的引擎接口,其中包 ...
- python和C语言混编的几种方式
Python这些年风头一直很盛,占据了很多领域的位置,Web.大数据.人工智能.运维均有它的身影,甚至图形界面做的也很顺,乃至full-stack这个词语刚出来的时候,似乎就是为了描述它. Pytho ...
- Python+C混编
Python最慢!C最快!Python+C混编?结果可想而知! 樱桃种子 百家号04-1712:11 共享库 使用C语言编译产生共享库,然后python使用ctype库里的cdll来打开共享库. 举例 ...
- Python和C|C++的混编(二):利用Cython进行混编
还能够使用Cython来实现混编 1 下载Cython.用python setup.py install进行安装 2 一个实例 ① 创建helloworld文件夹 创建helloworld.pyx,内 ...
- python扩展实现方法--python与c混和编程 转自:http://www.cnblogs.com/btchenguang/archive/2012/09/04/2670849.html
前言 需要扩展Python语言的理由: 创建Python扩展的步骤 1. 创建应用程序代码 2. 利用样板来包装代码 a. 包含python的头文件 b. 为每个模块的每一个函数增加一个型如PyObj ...
- python扩展实现方法--python与c混和编程
前言 需要扩展Python语言的理由: 创建Python扩展的步骤 1. 创建应用程序代码 2. 利用样板来包装代码 a. 包含python的头文件 b. 为每个模块的每一个函数增加一个型如PyObj ...
- swift objective-及c语言 混编
在xocde6出来我们大部分代码都是用objective-c写的(部分C/C++),现在出生来了一个新的语言叫swift,那么如何既能使用我们之前的代码,还可以使用新语言呢, 本文就此做一下说明. 关 ...
- [ios][swift]swift混编
http://blog.csdn.net/iflychenyang/article/details/8876542(如何在Objective-C的头文件引用C++的头文件) 1.将.m文件扩展名改为. ...
- Xcode中C、C++、Object-C3种语言的混编
转自: http://hi.baidu.com/onejw/item/f34390c997cdc226a1b50ae http://www.cocoachina.com/ask/questions/s ...
- Swift & OC 混编 浅析
转载自:http://www.infoq.com/cn/articles/wangyi-cartoon-swift-mixed-practice?utm_campaign=rightbar_v2&am ...
随机推荐
- 解决centos7 yum和wget都不能使用
1.前言 因为卸载了centos 7.9.2009 内置自带的python2.7.5,导致yum无法使用,但是又没有安装wget,两者不能相互安装导致死循环 2.安装wget 找到自己对应的系统版本, ...
- VMwarestation安装centos系统
上机实验1 1.软硬件准备 软件:推荐使用vmware 镜像:centos7下载网址https://mirror.iscas.ac.cn/centos/7/isos/x86_64/ 这个镜像网址实测比 ...
- 关于Powerlink和EtherCAT的对比
https://www.amobbs.com/thread-5679636-1-2.html 转发自:http://book.2cto.com/201508/55093.html 这个问题经常会被人问 ...
- playwright 实现高亮、is_visible 等源码修改
一. 前言 高亮是什么? 在ui自动化中可以执行js来让某个页面元素高亮,比如背景颜色.字体颜色.边框颜色等发生改变,以此更加方便执行的时候点了哪一步操作.(如果有不清楚怎么实现的可以转到我的介绍se ...
- JZOJ 4213. 【五校联考1day2】对你的爱深不见底
题目 思路 结论题,我不会证明: 找到第一个 \(|S_n| \leq m + 1\),那么答案就是 \(m - |S_{n-2}|\) 证明?我说了我不会,就当结论用吧 这已经很恶心了 然而这题还要 ...
- Cobalt Strike 之:域内渗透
郑重声明: 本笔记编写目的只用于安全知识提升,并与更多人共享安全知识,切勿使用笔记中的技术进行违法活动,利用笔记中的技术造成的后果与作者本人无关.倡导维护网络安全人人有责,共同维护网络文明和谐. Co ...
- openwrt 网络检测脚本
背景 openwrt 有些固件不太稳定,会时不时的断网,导致家里无法上网,遇到这种情况只能手动重启openwrt设备,该操作不方便,作为一个极客爱好者,那肯定是要实现自动化处理的了,写一个简单的脚本加 ...
- Solon2 的应用生命周期
Solon 框架的应用生命周期包括:一个初始化函数时机点 + 六个事件时机点 + 两个插件生命时机点 + 两个容器生命时机点(v2.2.0 版本的状态): 提醒: 启动过程完成后,项目才能正常运行(启 ...
- 2373. 矩阵中的局部最大值 (Easy)
问题描述 2373. 矩阵中的局部最大值 (Easy) 给你一个大小为 n x n 的整数矩阵 grid . 生成一个大小为 (n - 2) x (n - 2) 的整数矩阵 maxLocal ,并满足 ...
- VMwareSDN环境配置之git安装
------------恢复内容开始------------ null ------------恢复内容结束------------