python3调用C++

"本文介绍了如何使用Python3的ctypes模块调用C++动态库,通过数据类型的映射实现两者之间的交互。示例中展示了通过C++程序生成JSON字符串并传递给Python,以及需要注意的内存管理和数据格式问题。同时,提醒注意C++函数声明的extern "C"和jsoncpp版本的选择。"
摘要由CSDN通过智能技术生成


  python3可以通过ctypes模块调用C++库,适当地混合使用将融合python和C++的优势,提升程序的性能。
  在ubuntu下,python3通过ctypes模块调用C++动态库,其中的关键在于数据的转换,在python3和C++之间通过ctypes模块将它们的数据联系在一起,在 官网资料中,可支持的数据类型如下:

ctypes类型c类型python类型
c_bool_Boolbool
c_charchar单字符字节对象
c_wcharwchar_t但字符字符串
c_bytechar整型
c_ubyteunsigned char整型
c_shortshort整型
c_ushortunsigned short整型
c_intint整型
c_uintunsigned int整型
c_longlong整型
c_ulongunsigned long整型
c_ulonglongunsigned __int64或unsigned long long整型
c_size_tsize_t整型
c_ssize_tssize_t或Py_ssize_t整型
c_floatfloat浮点数
c_doulbedouble浮点数
c_longdoublelong double浮点数
c_char_pchar *字节串对象或None
c_wchar_pwchar_t *字节串或None
c_void_pvoid *int或None

  在上述数据结构中,可以看到只包含了基础的数据类型,如果传递大量的数据,可通过c_type_p传递字符串的方式进行,也可定义一些结构体传递数据。以下将用一个简单例子介绍通过传递json格式的字符串,程序下载链接为:https://gitee.com/jdi-shen/python-c

主要C++程序讲解

char* json_string() {
    Json::Value root;
    root["msg"] = "msg";
    root["x"] = 2.5;
    //std::string json_str = root.toStyledString();    //将json格式数据转变成字符串,程序将会出错
    std::string json_str = Json::FastWriter().write(root);    //将json格式数据转变成字符串

    //如果通过const_cast将string转变成char*,python将得到乱码
    static std::pair<char*, uint> c("", 0);    //first表示字符串数据,second表示字符串的大小
    if(c.second != json_str.size()) {    //如果与上次字符串数据的长度不同,则释放内存,避免内存泄漏
        if(c.second != 0) {    //如果字符串的长度为0,则不需要释放内存
            delete c.first;
        }
        c.first = new char(json_str.size());    //此处将发生内存泄漏的情况
        c.second = json_str.size();
    }
    for(int i = 0; i < json_str.size(); i++) {    //将string转变成char*
        c.first[i] = json_str[i];
    }

    return c.first;
}

python3程序

# -*- coding: utf-8 -*-
import ctypes
# 指定动态链接库
lib = ctypes.cdll.LoadLibrary('../lib/libcpp_lib.so')

#将python类型转换成c类型,支持int, float,string的变量和数组的转换
def convert_type(input):
    ctypes_map = {int:ctypes.c_int,
              float:ctypes.c_double,
              str:ctypes.c_char_p
              }
    input_type = type(input)
    if input_type is list:
        length = len(input)
        if length==0:
            print("convert type failed...input is "+input)
            return null
        else:
            arr = (ctypes_map[type(input[0])] * length)()
            for i in range(length):
                arr[i] = bytes(input[i],encoding="utf-8") if (type(input[0]) is str) else input[i]
            return arr
    else:
        if input_type in ctypes_map:
            return ctypes_map[input_type](bytes(input,encoding="utf-8") if type(input) is str else input)
        else:
            print("convert type failed...input is "+input)
            return null


import json
import time

if __name__ == '__main__':
    #需要指定返回值的类型,默认是int
    lib.add.restype = ctypes.c_double
    print(lib.add(convert_type(3.0), convert_type(2.0)))

    lib.json_string.restype = ctypes.c_char_p
    print(lib.json_string())

    # 由于数据格式不正确,出现乱码情况
    lib.string_test.restype = ctypes.c_char_p
    print(lib.string_test())

注意事项

  1. C++中的函数需要在声明处加入“extern “C” ”;
  2. jsoncpp的版本非常多,推荐使用gitee上的版本:https://gitee.com/openharmony/third_party_jsoncpp.git;
  3. python通过上述方式获得字符串后,会在字符串末尾增加一些不预期的字符,需要清除后再进行解析;
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值