• 背景介绍
    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混编)的更多相关文章

  1. python 与 matlab 混编

    用于 Python 的 MATLAB 引擎 API 快速入门 安装用于 Python 的 MATLAB 引擎 API Matlab的官方文档中介绍了 Matlab 与其余编程语言之间的引擎接口,其中包 ...

  2. python和C语言混编的几种方式

    Python这些年风头一直很盛,占据了很多领域的位置,Web.大数据.人工智能.运维均有它的身影,甚至图形界面做的也很顺,乃至full-stack这个词语刚出来的时候,似乎就是为了描述它. Pytho ...

  3. Python+C混编

    Python最慢!C最快!Python+C混编?结果可想而知! 樱桃种子 百家号04-1712:11 共享库 使用C语言编译产生共享库,然后python使用ctype库里的cdll来打开共享库. 举例 ...

  4. Python和C|C++的混编(二):利用Cython进行混编

    还能够使用Cython来实现混编 1 下载Cython.用python setup.py install进行安装 2 一个实例 ① 创建helloworld文件夹 创建helloworld.pyx,内 ...

  5. python扩展实现方法--python与c混和编程 转自:http://www.cnblogs.com/btchenguang/archive/2012/09/04/2670849.html

    前言 需要扩展Python语言的理由: 创建Python扩展的步骤 1. 创建应用程序代码 2. 利用样板来包装代码 a. 包含python的头文件 b. 为每个模块的每一个函数增加一个型如PyObj ...

  6. python扩展实现方法--python与c混和编程

    前言 需要扩展Python语言的理由: 创建Python扩展的步骤 1. 创建应用程序代码 2. 利用样板来包装代码 a. 包含python的头文件 b. 为每个模块的每一个函数增加一个型如PyObj ...

  7. swift objective-及c语言 混编

    在xocde6出来我们大部分代码都是用objective-c写的(部分C/C++),现在出生来了一个新的语言叫swift,那么如何既能使用我们之前的代码,还可以使用新语言呢, 本文就此做一下说明. 关 ...

  8. [ios][swift]swift混编

    http://blog.csdn.net/iflychenyang/article/details/8876542(如何在Objective-C的头文件引用C++的头文件) 1.将.m文件扩展名改为. ...

  9. Xcode中C、C++、Object-C3种语言的混编

    转自: http://hi.baidu.com/onejw/item/f34390c997cdc226a1b50ae http://www.cocoachina.com/ask/questions/s ...

  10. Swift & OC 混编 浅析

    转载自:http://www.infoq.com/cn/articles/wangyi-cartoon-swift-mixed-practice?utm_campaign=rightbar_v2&am ...

随机推荐

  1. 【DS】2.1

    线性表:c++各种基础操作里面,有&没&避免错的代码额~ #include <stdio.h> void test(int &x){//没有&就没有带回主函 ...

  2. 学习Java Day25

    今天学习了类路径和如何设置类路径 其中需要创建JAR后面才会介绍如何创建JAR,类路径可以节省空间并改善性能,设置类路径可以用-classpath选项指定路径

  3. 云小课|MRS数据分析-通过Spark Streaming作业消费Kafka数据

    阅识风云是华为云信息大咖,擅长将复杂信息多元化呈现,其出品的一张图(云图说).深入浅出的博文(云小课)或短视频(云视厅)总有一款能让您快速上手华为云.更多精彩内容请单击此处. 摘要:Spark Str ...

  4. JZOJ 3527.迷宫花坛(garden)

    题面 思路 考场想到 \(tarjan\) 缩点 然而忘了缩点怎么打 于是甩了个暴力 改题时学了个圆方树 发现挺好用 于是······注意重边 \(Code\) #include<cstdio& ...

  5. IP地址后面/24/26/27/28/29/30网关数量分别是多少?如何计算?

    转载csdn: https://blog.csdn.net/jinfengyunIDC/article/details/112575286

  6. C# 时间各种格式

    1.1 取当前年月日时分秒 currentTime=System.DateTime.Now; 1.2 取当前年 int 年=currentTime.Year; 1.3 取当前月 int 月=curre ...

  7. Java 文本检索神器 "正则表达式"

    Java 文本检索神器 "正则表达式" 每博一文案 在我们短促而又漫长的一生中,我们在苦苦地寻找人生的幸福,可幸福往往又与我们失之交臂, 当我们为此而耗尽宝贵的.青春年华,皱纹也悄 ...

  8. C#/VB.NET 如何在 Word 文档中添加页眉和页脚

    页眉位于文档中每个页面的顶部区域,常用于显示文档的附加信息,可以插入时间.图形.公司微标.文档标题.文件名或作者姓名等:页脚位于文档中每个页面的底部的区域,常用于显示文档的附加信息,可以在页脚中插入文 ...

  9. react的useRef

    在使用RN动画的时候,看到这样的代码: const App = () => { const fadeAnim = useRef(new Animated.Value(0)).current; / ...

  10. Kali配置gmssl密码算法库

    Kali配置gmssl密码算法库 一.密码算法库的下载 https://github.com/guanzhi/GmSSL/releases 二.安装配置 1 解压 把刚刚下载的GmSSL 3.0.0. ...