Boost(2):boost.python库介绍及简单示例

1. boost.python介绍

将c/c++的函数接口转换为Python接口有好几种解决方案,不同于C语言的简单直接,C++因使用了大量的面向对象编程的思想导致转换为Python接口时相对比较复杂,boost.python的目标就是为了简单方便地将C++程序转换成Python的接口。

BoostPython库是boost c++库的其中一个子库,使用它可以轻松地将C++的函数接口转换成Python接口。在大部分情况下你不需要对原先的C++代码做任何修改,boost.python会对C++类再做一层封装,使它编译后符合Python的语言规范。

2. 简单例程

按照程序员的江湖规矩,一个新的学习开始时,首先从HelloWorld程序下手。在这个例程中,首先使用c++函数输出"hello world",然后将这个函数封装成python的接口,在python环境下调用该函数实现打印。

注:本文例程运行在boost_1_73_0版本下。

整体文件结构如下:

/boost$ tree
├── 01_HelloWorld
│   ├── CMakeLists.txt
│   ├── hello.py
│   └── HelloWorld.cpp
└── CMakeLists.txt

1 directory, 4 files

创建boost目录用于保存后续的示例代码,boost目录下的CMakeLists.txt主要完成必要的库的查找工作,如这里会用到Python和Boost,然后再添加子目录下的CMakeLists.txt文件。


2.1 主目录下的CMakeLists.txt

内容如下:

project(Boost_Test)
cmake_minimum_required(VERSION 2.8.3)

# find python
find_package(PythonInterp REQUIRED)
find_package(PythonLibs ${PYTHON_VERSION_MAJOR}.${PYTHON_VERSION_MINOR} EXACT REQUIRED)

# now search for the boost component
# depending on the boost version it is called either python,
# python2, python27, python3, python36, python37, ...
list(
  APPEND _components
    python${PYTHON_VERSION_MAJOR}${PYTHON_VERSION_MINOR}
    python${PYTHON_VERSION_MAJOR}
    python
  )

message(BOOST_ROOT " ${BOOST_ROOT}")

set(_boost_python_found "")
set(Boost_NAMESPACE "libboost")
set(Boost_USE_MULTITHREADED ON)
set(Boost_USE_STATIC_LIBS OFF)
set(Boost_USE_STATIC_RUNTIME OFF)
foreach(_component IN ITEMS ${_components})
  find_package(Boost COMPONENTS ${_component})
  if(Boost_FOUND)
    set(_boost_python_found ${_component})
    break()
  endif()
endforeach()

#if(_boost_python_found STREQUAL "")
#  message(FATAL_ERROR "No matching Boost.Python component found")
#endif()

include_directories("${PYTHON_INCLUDE_DIRS}")
include_directories("${Boost_INCLUDE_DIRS}")
message(PYTHON_INCLUDE_DIRS " ${PYTHON_INCLUDE_DIRS}")
message(PYTHON_LIBRARIES " ${PYTHON_LIBRARIES}")
message(Boost_INCLUDE_DIRS " ${Boost_INCLUDE_DIRS}")
message(Boost_LIBRARIES " ${Boost_LIBRARIES}")

ADD_SUBDIRECTORY(01_HelloWorld)


2.2 HelloWorld.cpp

char const* greet()
{
   return "hello, world";
}

#include <boost/python.hpp>

BOOST_PYTHON_MODULE(hello)
{
    using namespace boost::python;
    def("greet", greet);
}


2.3 hello.py

#!/usr/bin/env python

import hello
print (hello.greet())


2.4 子目录下的CMakeLists.txt

这个文件主要完成编译库文件以及链接工作,注意MODULE_NAME必须与BOOST_PYTHON_MODULE()下的相同,比如这里都是hello。

set(MODULE_NAME  hello)

include_directories(${CMAKE_SOURCE_DIR})

add_library(${MODULE_NAME} SHARED
	HelloWorld.cpp
	)

if (UNIX)
  set_target_properties(${MODULE_NAME}
    PROPERTIES
    PREFIX ""
  )
elseif (WIN32)
  set_target_properties(${MODULE_NAME}
  PROPERTIES
  SUFFIX ".pyd"
  )
endif()

target_link_libraries(${MODULE_NAME}
  ${Boost_LIBRARIES}
  ${PYTHON_LIBRARIES}
)

对于windows系统,编译的python模块应该以".pyd"为后缀。而对于linux系统,默认会在模块名称前加"lib",而Boost.Python要求模块名称和库文件名称保持一致,所以这里要声明不对库文件添加前缀。

3. 编译运行

3.1 编译

cd boost
mkdir build
cmake ..
make

之后,在build/01_HelloWorld目录下会生成hello.so的库文件。

3.2 运行

cd 01_HelloWorld

# 在不install so的情况下,将python文件拷贝到so目录
cp ../../01_HelloWorld/hello.py .

python hello.py


3.3 运行结果

#python hello.py
hello, world


4. Boost接口

4.1 BOOST_PYTHON_MODULE宏

这个宏定义在boost/python/module.hpp文件下,该文件提供了创建Boost.Python扩展模块的基本工具。目前也只有这个宏。

BOOST_PYTHON_MODULE用于声明python模块初始化函数,其使用格式如下:

BOOST_PYTHON_MODULE(name)
{
   ...
}

name 参数必须与要初始化的模块的名称完全匹配,并且必须符合 Python 的标识符命名规则。

4.2 def

def接口是Boost.Python提供的自由函数版本,它能将当前作用域中的函数转换为python的接口,如示例中的greet()函数。后面还会看到有转换类内函数的版本。

def的使用方式如下:

#include <boost/python.hpp>

BOOST_PYTHON_MODULE(my_module)
{
	def("name", function_ptr);
	def("name", function_ptr, call_policies);
	def("name", function_ptr, "documentation string");
	def("name", function_ptr, call_policies, "documentation string");
}

其中name表示python接口下的函数名称,可以与c++的名称不同。function_ptr是对应的c++函数指针,即函数名。call_policies在后面用到的时候再详细说明。


参考资料

https://wiki.python.org/moin/boost.python/GettingStarted

https://www.boost.org/doc/libs/1_78_0/libs/python/doc/html/tutorial/index.html

https://www.boost.org/doc/libs/1_78_0/libs/python/doc/html/reference/high_level_components/boost_python_module_hpp.html

  • 8
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
使用Boost::Python可以非常方便地实现C++调用Python接口。以下是简单的步骤: 1. 安装Boost库和Python解释器。 2. 编写一个Python模块,在其中定义一些函数或类,这些函数或类是你希望C++调用的接口。 3. 用Boost::Python库Python模块导出到C++中,使得C++可以调用Python模块中的函数和类。 4. 在C++代码中调用Python模块中的函数或类。 下面是一个简单示例代码,演示了如何使用Boost::Python实现C++调用Python接口: ```cpp #include <boost/python.hpp> #include <iostream> // 定义一个Python函数 int add(int x, int y) { return x + y; } // 导出Python函数到C++ BOOST_PYTHON_MODULE(example) { using namespace boost::python; def("add", add); } int main() { // 初始化Python解释器 Py_Initialize(); // 导入Python模块 boost::python::object example_module = boost::python::import("example"); // 调用Python函数 boost::python::object result = example_module.attr("add")(1, 2); // 将Python返回值转换为C++类型 int sum = boost::python::extract<int>(result); // 输出结果 std::cout << "1 + 2 = " << sum << std::endl; // 释放Python解释器 Py_Finalize(); } ``` 在这个例子中,我们定义了一个名为`add`的Python函数,在C++中通过`def`函数将其导出。然后,在C++代码中导入了Python模块,调用了Python函数,并将其返回值转换为C++类型。最后输出了结果。 这只是一个简单的例子,Boost::Python还支持更复杂的数据类型和类的导出。如果你想深入了解Boost::Python的使用,可以参考官方文档和示例代码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

翔底

您的鼓励将是我创作最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值