pybind11以及打包学习

2024-06-04 17:38
文章标签 学习 打包 pybind11

本文主要是介绍pybind11以及打包学习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

pybind11以及打包学习

前言

最近在看fasttext,看到使用pybind11把c++代码封装了一下,然后打包后安装,python可以直接调用,非常方便,有点兴趣,手动试了简单例子,本篇没啥干货,简单记录下实现过程。

一 pybind11

c/c++代码都是用pybind11封装,可以直接用pip安装即可,官方给出的入门示例十分简单:

#include <pybind11/pybind11.h>int add(int i, int j) {return i + j;
}
PYBIND11_MODULE(example, m) {m.doc() = "pybind11 example plugin"; // optional module docstringm.def("add", &add, "A function which adds two numbers");
}

其中,add函数为c函数,也是需要导出的函数功能。PYBIND11_MODULE功能就是负责导出模块,第一个为宏参数(不需要双引号,setup.py中有参数对应),第二个参数就代表模块对象(py::module)。m.def负责定义需要导出的函数。官网强调,pybind是头文件依赖,即只需要指定头文件即可,不需要其它类库。

这样看,功能封装确实挺简单,下载官网上提供的python_example示例,直接进入根目录,用pip install . ,编译安装都搞定,然后运行示例中的测视例test.py, 感受一下使用的情况:

import python_example as massert m.__version__ == '0.0.1'
assert m.add(1, 2) == 3
assert m.subtract(1, 2) == -1

这里说下示例,编译的时候有一行会报语法错误:

m.attr("__version__") = VERSION_INFO;

这里应该是编译器认为用double类型给字符串赋值了,所以总是报错,搜了一下转字符串的宏:

#define STR1(R)  #R
#define STR2(R) STR1(R)

然后调用STR2转换一下就行了。

二 打包

打包主要借助setuptools提供的接口,需要的操作都在setup.py中定义,然后用pip安装。还是拿python_example当例子:

ext_modules = [Extension('python_example',# Sort input source files to ensure bit-for-bit reproducible buildssorted(['src/main.cpp']),include_dirs=[# Path to pybind11 headersget_pybind_include(),],language='c++'),
]

这个就是定义需要导出模块的一些信息,包括模块宏名,前面提到的PYBIND11_MODULE的第一个参数就是对应的Extension的第一个参数的名字,如果两者不一致,会导致编译错误;第二个参数定义了需要编译的源文件; include_dirs是依赖的头文件;language指定源语言。

class BuildExt(build_ext):"""A custom build extension for adding compiler-specific options."""def build_extensions(self):for ext in self.extensions:ext.define_macros = [('VERSION_INFO', '{}"'.format(self.distribution.get_version()))]ext.extra_compile_args = optsext.extra_link_args = link_optsbuild_ext.build_extensions(self)

这个就是自定义模块编译地方,对于每个自定义的模块(即前面的ext_modules),在build_extensions函数中,可以对每个模块指定一些宏定义(define_macros)、编译参数(extra_compile_args)以及链接参数(extra_link_args)等, 用于模块编译的需要。

而最后一部分setup对象:

setup(name='python_example',version=__version__,author='Sylvain Corlay',author_email='sylvain.corlay@gmail.com',url='https://github.com/pybind/python_example',description='A test project using pybind11',long_description='',ext_modules=ext_modules,setup_requires=['pybind11>=2.5.0'],cmdclass={'build_ext': BuildExt},zip_safe=False,
)

定义了模块的一些信息,如模块名(import时使用),作者,版本号之类的,其中需要关注的是cmdclass参数,这里指定了模块编译需要使用的类。整个setup.py核心内容就这些,是不是感觉也不复杂?

对了,有的同学可能会问,c/c++编译器好像没指定哦?是的,这个交给setuptools去干了,官网介绍是setuptools会找到当初安装python的那套环境去编译这些模块,也就是既然python都装上了,默认你的机器上编译环境应该是可以编译这些自定义模块的。

总结

总体来说,使用起来感觉挺方便的。但是对我来讲,最起码暴露了一个问题,c/c++好久没用了,上面那个示例编译错误,搞挺久才搞定,技艺生疏了。另外,更底层的流程,还是一窍不通,比如,python函数调用是怎么调用到c/c++库中去的,模块到底是怎么定义的(我没找到编译出来的库文件),有时间再继续扒吧。

附录

pybind11源码

pybind11官方文档

python_example源码

这篇关于pybind11以及打包学习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030673

相关文章

Python项目打包部署到服务器的实现

《Python项目打包部署到服务器的实现》本文主要介绍了PyCharm和Ubuntu服务器部署Python项目,包括打包、上传、安装和设置自启动服务的步骤,具有一定的参考价值,感兴趣的可以了解一下... 目录一、准备工作二、项目打包三、部署到服务器四、设置服务自启动一、准备工作开发环境:本文以PyChar

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall

javafx 如何将项目打包为 Windows 的可执行文件exe

《javafx如何将项目打包为Windows的可执行文件exe》文章介绍了三种将JavaFX项目打包为.exe文件的方法:方法1使用jpackage(适用于JDK14及以上版本),方法2使用La... 目录方法 1:使用 jpackage(适用于 JDK 14 及更高版本)方法 2:使用 Launch4j(

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

springboot3打包成war包,用tomcat8启动

1、在pom中,将打包类型改为war <packaging>war</packaging> 2、pom中排除SpringBoot内置的Tomcat容器并添加Tomcat依赖,用于编译和测试,         *依赖时一定设置 scope 为 provided (相当于 tomcat 依赖只在本地运行和测试的时候有效,         打包的时候会排除这个依赖)<scope>provided

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss