当前位置: 首页 > news >正文

C++/Python混合编程之Pybind11的使用

一、pybind11简单介绍

一些python和C++混编方案:

  1. Python官方的Python/C API, 缺点:所有数据类型必须手动改为Cpython封装的binding类型;
  2. Cython,编译器支持python代码转为C代码,缺点:移值和复用成本高
  3. SIWG 主要解决高级语言与C和C++语言交互问题,支持10几种编程语言如java、python、C#等。在python端性能表现不太好
  4. Boost.Python  为C++中广泛应用的Boost开源库,编译和依赖关系繁重,如果只用于解决python的交互,大材小用了。
  5. pyind11   理解为Boost.Python的蓝本,基于C++11应用了很多新特性,Pybind11 通过 C++ 编译时的自省来推断类型信息,来最大程度地减少传统拓展 Python 模块时繁杂的样板代码, 且实现了常见数据类型,如 STL 数据结构、智能指针、类、函数重载、实例方法等到 Python 的自动转换,其中函数可以接收和返回自定义数据类型的值、指针或引用

简言之,Pybind11 是一个轻量级的 C++ 库,用于将你的 C++ 代码暴露给 Python 调用(反之也可,但主要还是前者)。Pybind11 借鉴了 Boost::Python 库的设计,但使用了更为简洁的实现方式,保证开发效率和实用性。其帮助文档链接如下:

官方文档:  pybind11 documentation

中文版本: 安装库 — pybind11 文档

二、****pybind11 环境准备

1.配置环境说明

window10+  VS2017+python3.10(本电脑安装了Anaconda是python3.10这个环境,也可以使用其它版本,window要求Visual Studio 2017 及更新版本)

2.安装pybind11

方式1:直接下载源代码 git clone GitHub - pybind/pybind11: Seamless operability between C++11 and Python

方式2:直接用pip命令安装:pip install pybind11  或使用 pip3 install pybind11 (pip3同时会下载python3.0+版本)
image

查看python -m 默认下载路径:
image
        直接指定存放目录的下载方式:

python -m pip install pybind11 --target=D:\python\site-packages

下载的pybind11目录如下:

image

三、python调用C++动态库****测试例子

底层逻辑:把C++中编写的函数,用pybind封装为调用的pyd库,成功在python中import调用;

说明:pybind11 是只包含头文件的库,因此不需要链接到任何特殊的库,也没有中间(magic)转换步骤。win系统上C++编程时只要包含到include路径就行。注意要包含对应的python库目录。

1.VS属性配置

VS2017上新建项目,属性中包含pybind 和python对应的include目录。 我使用的是conda中python3.10所以路径不一样。

image

附加库目录,增加python的库目录:

image

添加python的lib库:

image

在工程属性—>常规—>目标文件扩展名修改为.pyd,配置类型改为.dll

image

2.编写绑定文件并测试

方法:使用PYBIND11_MODULE()将所有需要封装的代码都放在里面。

PYBIND11_MODULE() 宏创建了一个函数,当 Python 内部发出 import 语句时将调用该函数。

第一个参数:模块名(testPyd)作为第一个宏参数给出(它不应该在引号中)。

第二个参数:m定义了类型的变量 py::module_ ,它是创建绑定的主接口。

module_::def() :会生成绑定代码,将 add() 函数公开给 Python。

CPP文件如下:

image

编译生成文件:(生成的文件要和绑定模块名称一致)
image

其中,pyd文件是生成的动态库,可供给python调用。以下为python调用testPyd.pyd的add函数的测试结果:

image

如图,在python中成功调用add函数。 注意运行时一定要在有testPyd.pyd文件的目录。

3.结构体类型的绑定

image

在python中测试如下:

image

上面print(p)的信息没有具体name,可以使用 __repr__函数:

image

image

4.其它数据类型

可以查看帮助文档,按照上面的方式绑定到模块中。如果C++文件中代码调用了第三方dll库,在python中调用时,需要把dll文件放在py文件运行目录中

四、python调用C++****的测试例子

底层逻辑:pybind11 使用C++ 封装了 Python 类型和函数,通过python解释器可以供C++调用

1.编写一个example.py文件:【我存放于工程目录子文件夹python_scripts中】

image

2.编写调用example模块的C++main文件:

image

image

3.运行测试main代码

记得在属性中把,输出文件格式改为.exe。测试打印结果如下:

image

结果一致!

4.遗留问题

在初始化python环境时,开始使用的是py::scoped_interpreter guard{};初始化方法,但这个方法报如下错误,有检测自己的环境配置都是正常,也按初始环境的打印设置了路径,仍未解决。原因不明,暂时搁置在这里:

引发了未经处理的异常:读取访问权限冲突。

this->m_ptr->**ob_type** 是 0xFFFFFFFFFFFFFFE7。
image

注:本文搬运自本人20250503CSDN文章:https://blog.csdn.net/sinat_26398509/article/details/146230196?spm=1001.2014.3001.5501

http://www.jsqmd.com/news/412580/

相关文章:

  • SRE 应用稳定性看板-从应用维度监控服务健康状态,基于 Apdex 评分体系
  • 大数据领域数据中台的质量评估方法
  • 使用 Terraform + Terragrunt 管理 AWS 基础设施项目说明
  • **4皇后问题回溯搜索过程**的图文解析、关键函数说明及核心考点总结,结构清晰、逻辑准确
  • 系统思考:自由职业背后的悖论
  • Sora2 免费去水印网站
  • **回溯法在两个经典问题(0-1背包、n皇后)中的应用**的清晰解读,涵盖了搜索树结构、剪枝策略、可行解识别与核心约束条件
  • Learning on the Manifold: Unlocking Standard Diffusion Transformers withRepresentation Encoders
  • **分支限界法(结合回溯思想)求解0-1背包问题**的核心流程与结果
  • 20260225 之所思 - 人生如梦
  • build_fsd_luyan_from_rm——注释
  • 回溯法的两种实现方式(迭代与递归)本质上都是对解空间树进行深度优先搜索(DFS),区别在于控制搜索过程的机制不同
  • WPF implement DelCommand inherited from ICommand from scratch
  • **0-1 背包问题的分支限界法(Branch and Bound)求解框架**,核心融合了**贪心松弛上界估计**与**精确剪枝策略**
  • N9e配置电话告警,实现故障的电话(语音)通知
  • Grafana + Loki 使用说明
  • windows上子系统WSL下载和使用
  • linux系统 Qt 通常的目录结构
  • 算法备案分类详细及合规要点整理(2026年更新版)
  • MySQL数据库从win导出成_db.sql复制到linux
  • EC2 使用 dnsmasq 本地缓存 + EKS 使用 NodeLocalDNS
  • 基于 Kubernetes + Helm 部署高可用 ETCD 集群
  • OS 核心知识点全解析(一)
  • Redis 迁移方案-RedisShake
  • qml可拖动折线图
  • 【linuxqt】qsql_mysql.cpp:57:10: fatal error: QtSql/private/qsqldriver_p.h: No such file or directory
  • 我草我怎么这么牛
  • 基于 AWS Global Accelerator 实现全球低延迟访问-RapidX 全球加速方案
  • day96(2.25)——leetcode面试经典150
  • 【Linux】进程的页表详解