python中调用C语言动态库的方法,以及指针、引用、数组指针、结构体指针、字符串等参数的传递(新增回调函数用法)

本文详细介绍了如何在Python中使用ctypes模块来操作C语言动态库,包括传递不同类型的参数如指针、引用、数组指针、结构体指针和字符串,并展示了如何处理函数返回值。此外,还涉及了回调函数的使用,以及错误码处理函数的注册。
摘要由CSDN通过智能技术生成

原文连接:python中ctypes使用:指针和指针数组的调用

目录

导入C语言动态库

参数传递问题(指针、引用、数组指针、结构体指针、字符串)

接收函数的非空返回值的问题


导入C语言动态库

# coding=utf-8
import os

os.add_dll_directory("动态库所在的绝对路径")
test = cdll.LoadLibrary('动态库名称.dll')

参数传递问题(指针、引用、数组指针、结构体指针、字符串)

# coding=utf-8
import os
import ctypes
from ctypes import *

os.add_dll_directory("动态库所在的绝对路径")
test = cdll.LoadLibrary('动态库名称.dll')

## 传递指针
# 定义一个指向c_int的指针
# c_int(0) 代表的是这个c_int变量的值为0,即c_int(6)代表变量值为6,而不是申请了一个拥有6个c_int变量的数组
pValue = pointer(c_int(0))
# C语言函数原型:void FunPoint(int *pValue);
test.FunPoint(pValue)


## 传递引用
value = c_int(0)
# C语言函数原型:void FunByref(int &value);
test.FunByref(byref(value))


## 传递数组指针
# data:一个指向(拥有6个c_int类型的元素的数组)的指针
data = (c_int * 6)()
# C语言函数原型:void FunArray(int *arr, int size);
test.FunArray(data, 6)


## 传递结构体指针
class StructureTmp(Structure):
    _fields_ = [
        ('name', c_int),
        ('age', c_int),
    ]

# data2:一个指向(拥有6个结构体 StructureTmp 类型的元素的数组)的指针
data2 = (StructureTmp * 6)()
# C语言函数原型:void FunStructArray(StructureTmp *arr, int size);
Struct_Ptr = ctypes.POINTER(StructureTmp)
# 约定函数的传入参数的类型
test.FunStructArray.argtypes = [Struct_Ptr, c_int]
test.FunStructArray(data2, c_int(6))


## 传递字符串
snSize = 64
# 下面两种方法都可以
sn = (c_char * snSize)()
sn = create_string_buffer(''.encode('utf-8'), snSize)
# C语言函数原型:void FunStr(char *pStr, int size);
test.FunStr(sn, c_int(snSize))
# 输出字符串
print('GetSn = ' + str(sn.value))

接收函数的非空返回值的问题

# coding=utf-8
import os
import ctypes
from ctypes import *

os.add_dll_directory("动态库所在的绝对路径")
test = cdll.LoadLibrary('动态库名称.dll')


## 约定函数的返回值的类型
# C语言函数原型:int Fun1();
test.Fun1.restype = ctypes.c_int
ret = test.Fun1()


## 函数返回char *(字符串)
# C语言函数原型:char * FunStr();
test.FunStr.restype = ctypes.c_char_p
print("str :" + str(test.FunStr()))

 回调函数

# 定义错误码处理函数
def PrintErrorCode(errorCode, axisIndex):
    print('get callback req:', errorCode + axisIndex)


# 定义错误码处理函数的 函数类型(返回值和传入参数)
# 第一个None代表返回值为void
# 后面为参数函数的传入参数
ErrorCodeFun = CFUNCTYPE(None, c_int, c_int)


if __name__ == '__main__':
    PrintErrorCode(1, 2)

    # 获取错误码处理函数的变量
    errorCodeFun = ErrorCodeFun(PrintErrorCode)
    # 注册错误码处理函数
    device.SetHandleErrorCodeFun(errorCodeFun)

    # 以下为错误用法
    # 声明函数的传入参数类型,然后直接传入函数名这种会发生错误,原因未知
    device.SetHandleErrorCodeFun.argtypes = [ErrorCodeFun]
    device.SetHandleErrorCodeFun(PrintErrorCode)

  • 6
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Python可以通过ctypes库来实现C语言结构体的定义和使用。在ctypes库,可以使用Structure类来定义C语言结构体,并使用byref函数来获取结构体的指针。为了自动生成C语言结构体赋值函数,可以使用Python字符串格式化功能和反射机制来实现。具体步骤如下: 1. 定义C语言结构体,并使用Structure类来定义Python结构体。 2. 使用字符串格式化功能和反射机制来生成C语言结构体赋值函数的代码。 3. 将生成的代码写入到文件,即可实现自动生成C语言结构体赋值函数的功能。 下面是一个简单的例子,演示了如何使用Python自动生成C语言结构体赋值函数的代码: ``` import ctypes class Student(ctypes.Structure): _fields_ = [("name", ctypes.c_char_p), ("age", ctypes.c_int), ("score", ctypes.c_double)] def generate_struct_assign_func(struct_name): struct_type = getattr(ctypes, struct_name) fields = [f[0] for f in struct_type._fields_] func_name = "assign_{}".format(struct_name.lower()) func_args = ", ".join(["{}={}".format(f, f) for f in fields]) func_code = "void {}({} *s) {{\n".format(func_name, struct_name) for f in fields: func_code += " s->{} = {};\n".format(f, f) func_code += "}\n" return func_code if __name__ == "__main__": struct_name = "Student" func_code = generate_struct_assign_func(struct_name) with open("{}.c".format(struct_name.lower()), "w") as f: f.write(func_code) ``` 这个例子,我们定义了一个名为Student的C语言结构体,并使用Structure类来定义Python结构体。然后,我们定义了一个名为generate_struct_assign_func的函数,用于生成C语言结构体赋值函数的代码。最后,我们调用generate_struct_assign_func函数,并将生成的代码写入到文件。运行这个程序后,会在当前目录下生成一个名为student.c的文件,其包含了自动生成的C语言结构体赋值函数的代码。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值