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

相关文章

Flutter打包APK的几种方式小结

《Flutter打包APK的几种方式小结》Flutter打包不同于RN,Flutter可以在AndroidStudio里编写Flutter代码并最终打包为APK,本篇主要阐述涉及到的几种打包方式,通... 目录前言1. android原生打包APK方式2. Flutter通过原生工程打包方式3. Futte

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

linux打包解压命令方式

《linux打包解压命令方式》文章介绍了Linux系统中常用的打包和解压命令,包括tar和zip,使用tar命令可以创建和解压tar格式的归档文件,使用zip命令可以创建和解压zip格式的压缩文件,每... 目录Lijavascriptnux 打包和解压命令打包命令解压命令总结linux 打包和解压命令打

将java程序打包成可执行文件的实现方式

《将java程序打包成可执行文件的实现方式》本文介绍了将Java程序打包成可执行文件的三种方法:手动打包(将编译后的代码及JRE运行环境一起打包),使用第三方打包工具(如Launch4j)和JDK自带... 目录1.问题提出2.如何将Java程序打包成可执行文件2.1将编译后的代码及jre运行环境一起打包2

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

配置springboot项目动静分离打包分离lib方式

《配置springboot项目动静分离打包分离lib方式》本文介绍了如何将SpringBoot工程中的静态资源和配置文件分离出来,以减少jar包大小,方便修改配置文件,通过在jar包同级目录创建co... 目录前言1、分离配置文件原理2、pom文件配置3、使用package命令打包4、总结前言默认情况下,

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