IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> Python如何调用C和C++ -> 正文阅读

[C++知识库]Python如何调用C和C++

ctypes库简介

根据Python官方文档,ctypes是一个外部函数库,它提供了与C兼容的数据类型,允许调用DLL(Dynamic Link Libraries, 动态链接库)或共享库中的函数。换句话说,通过ctypes库,我们能在Python程序中调用C/C++代码。

动态链接库是一个已编译的二进制文件,其在程序编译时并不会被链接到目标代码,而是在程序运行时才载入。Windows上的动态链接库为DLL(.dll),Linux上为SO(.so

示例演示

先给一个演示的Demo,然后再展开来讲,演示环境说明如下:

Platform: Ubuntu-20.04
gcc: 9.4.0

首先新建一个名为test.cpp的源码文件,源码内容如下:

extern "C"{
    void greet(char* name){
        std::cout << "Hello " << name << std::endl;
    }
    
    int sumArray(int a [], int n){
        int s = 0;
        for (int i = 0; i < n;i++){
            s += a[i];
        }
        return s;
    }

    double distance(double *x, double y[], int n){
        double dis;
        for (int i = 0; i < n;i++){
            dis += pow((x[i] - y[i]), 2);
        }
        return sqrt(dis);
    }
}

然后通过下述命令将其编译为动态链接库:

g++ -fPIC -shared test.cpp -o test.so

Python调用上述三个函数的示例如下:

from ctypes import *

# 加载
lib = CDLL("./test.so")

lib.greet(b"Tom")
# Hello Tom

int_5 = c_int * 5
arr = int_5(1, 3, 5, 7, 9)
print(lib.sumArray(arr, 5))
# 25

double_2 = c_double * 2
x = double_2(1, 3)
y = double_2(2, 4)
distance = lib.distance
distance.restype = c_double
print(distance(x, y, 2))
# 1.4142135623730951

C/C++部分

从上述演示示例可以看出,对于C/C++源码仅需要将其编译为动态链接库即可,上述的示例命令为:

g++ -fPIC -shared test.cpp -o test.so

说明:

  • 若为C程序使用gcc,若为C++程序则使用g++
  • -fPIC表示位置独立。
  • -shared表示编译为动态库。

另外,需要注意在上述示例代码中还使用了extern "C" {}的用法,其作用是告知编译器按C的方式编译,编译后可以直接通过函数名调用。这在编译C++程序时是必须的,因为C++支持函数重载,导致编译后函数名会发生改变,使得不能通过函数名来对C++程序中的函数进行调用。

注意:extern修饰代表本模块可以在外部使用,若不想暴露相应的接口,则可以使用static修饰。

Python部分

加载动态库

在Python中载入动态链接库的方式如下所示:

from ctypes import *

# 方式1
lib = CDLL("./test.so")

# 方式2
lib = cdll.LoadLibrary("./test.so")

载入完成后便可以通过.来调用C/C++程序中的内容。

数据类型间的对应关系

ctypes中定义的与C兼容的基本数据类型完整版详见官网Fundamental data types。本文仅列举一些常用的:

ctypes类型C类型Python类型
c_bool_Bool布尔型
c_charchar单字符字节对象
c_intint整型
c_floatfloat浮点型
c_doubledouble浮点型(python不区分单精度还是双精度)

字符串

Python中通过ctypes传递参数类型为字符串的形式如下:

# 形式一
lib.travel(b"I love Python!")

# 形式二:使用字符指针
m_str = c_char_p(b"I love Python!")
lib.travel(m_str)

# 形式三
m_str = "我爱Python!"
lib.travel(m_str.encode())

# 形式四:创建String Buffer
m_str = create_string_buffer(("我爱Python!").encode())
lib.travel(m_str)

其中traval是C++中遍历打印字符串的一个函数,函数定义如下:

void travel(char * str){
    for(int i = 0;str[i];i++){
        std::cout << str[i];
    }
    std::cout << std::endl;
}

说明:

  • 加上b可以让字符串强制转为bytes类型,但这种方式仅限于只包含ASCII字符的字符串。
  • 若字符串中不仅包含ASCII字符,可以使用encoode()方法。
  • ctypes提供了函数create_string_buffer()来创建字符串缓冲区,其属于可修改的字符串传参方式

下面的示例便验证了create_string_buffer能创建可修改的字符串参数。

C++中修改函数定义:

void increment(char * str){
    for(int i = 0; str[i]; i++){
        str[i] += 1;
    }
}

Python中调用及结果如下:

m_str = create_string_buffer(b"abc")
lib.increment(m_str)
print(m_str.value)
# b'bcd'

指针/引用

指针可以通过ctypes中的pointer(obj)函数进行创建:

pi = c_float(3.14)
ptr = pointer(pi)
print(ptr.contents)
# c_float(3.140000104904175)

引用可以通过ctypes中的byref(obj)函数进行创建:

pi = c_float(3.14)
ptr = byref(pi)

注意:官网指明若只想向外部函数传递一个对象指针,使用引用更快

下面给出一个交换两个元素值的例子:

C++部分源码为:

void swap(int *a, int &b){
    int t = *a;
    *a = b;
    b = t;
}

Python部分源码为:

x = c_int(3)
y = c_int(4)
lib.swap(pointer(x), byref(y))
print(x.value, y.value)
# 4 3

数组类型

创建数组类型的推荐方式是类型乘以一个正数,例如:

int_3 = c_int * 3

传参

通过argtypes属性可以指定函数的传参类型。示例如下:

print_str = lib.travel
print_str.argtypes = [c_char_p]

返回值类型

默认情况下都假定函数返回c_int类型,但可以通过函数对象的restype属性可以指定返回值的类型。在上述示例演示中便有一个现成的例子,其指定了返回值类型是c_double

distance.restype = c_double

参考资料

完成本文参考了如下资料:

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-11-05 00:06:35  更:2022-11-05 00:08:52 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/19 3:17:16-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码