结构体示例:

这里是 C 代码的部分,主要是结构体的声明和回调函数定义。

// 新版本定义
typedef enum {
DevCard,
DevLocator,
DevReader
} DevType; typedef enum {
MsgLocate, // 定位信号
MsgCut, // 剪断信号
MsgHeartBeat, // 心跳信号
MsgCall, // 呼叫信号
MsgShake, // 震动信号
MsgLeave, // 离开信号
MsgOffAlarm, // 报警灯关闭信号
MsgCheckPower, // 电量检测信号
MsgHeartRate, // 心率信号
MsgExtra, // 补发信号
MsgNoPower, // 没电信号
MsgReset, // 重置信号
MsgBoundary, // 跨界信号
// 此消息会跟随着MsgLocate消息出现
MsgPower,
// 当checkOffline设置为1时有效。如果在offlineTime时间内没有收到此设备的信号,则发出离线信号
MsgOffline,
MsgLast
} MsgType; // 由于消息类型的不同,不是所有的值都有意义
typedef struct {
MsgType type;
DevType device; // 硬件类型
int32_t cardID; // 标签/腕带ID,当DevType为DevCard时有意义
int32_t readerID; // 阅读器ID,总是有意义
int32_t locatorID; // 大于0时有意义,表示定位器ID
int32_t readerRSSI; // 阅读器信号强度,由于硬件版本的差异,此值要么全有意义,要么全没意义(总是0)
int32_t locatorRssiX; // 当MsgType为[MsgLocate, MsgExtra]时,此值有意义,表示定位器的信号强度
int32_t locatorRssiY; // 定位器的信号强度
uint8_t isCut; // 如果此值为1,表示硬件被破坏。因为有专门的MsgCut消息,可以忽视此值
uint8_t isShake; // 如果此值为1,表示硬件处于震动状态。仅当MsgType为MsgShake是有意义,1表示震动,0表示静止
uint8_t isLowpower; // 硬件是否低电。硬件为标签时有意义,因为有追加的MsgPower消息,可以在MsgType为MsgPower时使用此值
int32_t heartRate; // 心率,当MsgType为[MsgHeartBeat, MsgHeartRate]之一时有意义
int32_t power; // 电量,当MsgType为[MsgHeartBeat, MsgReset, MsgCheckPower]之一时有意义
int64_t time; // 消息时间戳
int version; // 硬件版本号,DevType为DevCard时有意义
} DevMsg; typedef void(*MsgCallback)(DevMsg msg, void *ctx); extern "C" {
DEVSDK_API int ccrfidDevSdkStartWork(MsgCallback fun, void *ctx);
}
# coding=UTF-8
from ctypes import *
import time
import _ctypes # 定义回调函数参数的结构体
class DevMsg(Structure):
_fields_ = [("type", c_long),
("device", c_long),
("cardID", c_long),
("readerID", c_long),
("locatorID", c_long),
("readerRSSI", c_long),
("locatorRssiX", c_long),
("locatorRssiY", c_long),
("isCut", c_char),
("isShake", c_char),
("isLowpower", c_char),
("heartRate", c_long),
("power", c_long),
("time", c_longlong),
("version", c_long)]

注意一下 _fields_的内容:这里就是对 C 数据类型的转换。左边是 C 的结构成员名称,右边则是在 python 中声明一下各个成员的类型。其他的一些类型请参见官方文档。

此外还需要注意一下类似于 c_int, c_void_p 等等的定义是在 ctypes 中的,如果是用import ctypes 的方式包含 ctypes 模块,则应该写成 ctypes.c_long, ctypes.c_longlong,ctypes.c_char

第三个要注意的是:这个类必须定义为 ctypes.Structure 的子类,否则在进行后续的函数传递时,ctypes 由于不知道如何进行数据类型的对应,会抛出异常

回调函数示例:

# 定义功能类与方法
class DemoDll: def __init__(self):
self.dll = cdll.LoadLibrary('ccrfidDevSDK.dll')
return def ccrfidDevSdkStartWork(self, callback, p):
return self.dll.ccrfidDevSdkStartWork(callback, p) # ctypes通过 CFUNCTYPE 支持回调函数,定义返回值与参数,第一个参数表示返回值,void为None
CALLBACK = CFUNCTYPE(None, POINTER(DevMsg))

最后的 CALLBACK通过 ctypes 定义了一个回调函数类型,这个在后面的调用中需要使用

CFUNCTYPE 后面的第一个参数为 None,这表示回调函数的返回值类型为 void

完整功能示例:

# coding=UTF-8
from ctypes import *
import time
import _ctypes # 定义回调函数参数的结构体
class DevMsg(Structure):
_fields_ = [("type", c_long),
("device", c_long),
("cardID", c_long),
("readerID", c_long),
("locatorID", c_long),
("readerRSSI", c_long),
("locatorRssiX", c_long),
("locatorRssiY", c_long),
("isCut", c_char),
("isShake", c_char),
("isLowpower", c_char),
("heartRate", c_long),
("power", c_long),
("time", c_longlong),
("version", c_long)] # 定义功能类与方法
class DemoDll: def __init__(self):
self.dll = cdll.LoadLibrary('ccrfidDevSDK.dll')
return def ccrfidDevSdkStartWork(self, callback, p):
return self.dll.ccrfidDevSdkStartWork(callback, p) # ctypes通过 CFUNCTYPE 支持回调函数,定义返回值与参数,第一个参数表示返回值,void为None,第二参数为回调函数的参数为结构体指针
CALLBACK = CFUNCTYPE(None, POINTER(DevMsg)) # 回调函数距离功能实现
def _callback(para):
# print(dir(para))
obj = para.__getitem__(0)
print(obj.type, obj.cardID, obj.readerID, obj.locatorID, obj.heartRate, obj.power, obj.time, obj.version) # 定义回调函数
callBackFunc = CALLBACK(_callback) # 实例化功能类
dll = DemoDll()
# 定义一个空的指针
null_ptr = POINTER(c_int)()
# 具体功能调用
c = dll.ccrfidDevSdkStartWork(callBackFunc, null_ptr) print(c) time.sleep(100000)

运行结果

python 调用dll 动态链接库 结构体参数及回调函数等示例的更多相关文章

  1. Python调用DLL动态链接库——ctypes使用

    最近要使用python调用C++编译生成的DLL动态链接库,因此学习了一下ctypes库的基本使用. ctypes是一个用于Python的外部函数库,它提供C兼容的数据类型,并允许在DLL或共享库中调 ...

  2. python调用c/c++时传递结构体参数

    背景:使用python调用linux的动态库SO文件,并调用里边的c函数,向里边传递结构体参数.直接上代码 //test1.c # include <stdio.h> # include ...

  3. 绝对好文C#调用C++DLL传递结构体数组的终极解决方案

    C#调用C++DLL传递结构体数组的终极解决方案 时间 2013-09-17 18:40:56 CSDN博客相似文章 (0) 原文  http://blog.csdn.net/xxdddail/art ...

  4. C#中调用Dll动态链接库

    C#中调用Dll动态链接库 起始 受限于语言的不同,我们有的时候可能会用别人提供的函数及方法 或者其他的什么原因.反正就是要调!!! 恰巧别人所使用的的语言跟自己又不是一样的 这个时候想要调用别人的函 ...

  5. x264中重要结构体参数解释,参数设置,函数说明 <转>

    x264中重要结构体参数解释http://www.usr.cc/thread-51995-1-3.htmlx264参数设置http://www.usr.cc/thread-51996-1-3.html ...

  6. Qt--信号槽传递自定义结构体参数

    自定义结构体参数的信号槽连接 (1) 对于自定义的结构体参数,信号槽无法识别参数,导致信号槽连接不起作用.所以需要注册结构体参数.在结构体中声明结束的地方加上结构体注册. struct DealDet ...

  7. python调用dll详解

    参考链接https://www.cnblogs.com/TQCAI/p/8881530.html https://www.jb51.net/article/52513.htm https://www. ...

  8. Go结构体实现类似成员函数机制

    Go语言结构体成员能否是函数,从而实现类似类的成员函数的机制呢?答案是肯定的. package main import "fmt" type stru struct { testf ...

  9. (60) 结构体指针、结构体变量嵌套、结构体指针嵌套、函数指针、数组指针、指针数组、typedef 综合运用

    #include<stdio.h> #include<iostream> #include<malloc.h> /* author : 吴永聪 program: 结 ...

随机推荐

  1. 通过ecplise导入mysql的jar包时,右键找不到build path问题

    当我们执行java连接数据库程序的时候,我们需要再我们的项目里导入mysql的jar包,这时,我们需要右键->build path进行导入,但是有时候我们右键的时候并没有出现build path ...

  2. Bootstrap Table 的X-editable插件怎么用

    在准备使用X-editable来做Bootstrap Table 的行内编辑的时候,根据http://www.cnblogs.com/landea... 的文章,我不能将全部效果重复实现,网上也搜索了 ...

  3. 大v用户数据统计分析

    1,统计数据的基本情况,包括微博总数,用户总数,最活跃和最不活跃的用户id #!/bin/sh source_dir=/home/minelab/data/DATA source_file_name= ...

  4. 二、启动一款app演示

    一.下载aapt包 1. aapt即Android Asset Packaging Tool,在SDK的build-tools目录下.该工具可以查看apk包名和launcherActivity 2.打 ...

  5. RequestMapping 注解的解析、匹配、注册

    RequestMapping 注解的解析.匹配.注册 1)创建 RequestMappingHandlerMapping 实例时会触发 afterPropertiesSet 调用. 2)读取容器中所有 ...

  6. 硬件-硬盘-SSD(固态硬盘):百科

    ylbtech-硬件-硬盘-SSD(固态硬盘):百科 固态驱动器(Solid State Disk或Solid State Drive,简称SSD),俗称固态硬盘,固态硬盘是用固态电子存储芯片阵列而制 ...

  7. 那些年我们经历的BT面试题

    初入职场面试的我到处碰壁,以下是我个人对几道面试题的小总结: 1.一列数字的规则如下:1,1,2,3,5,8,13,21, 34........ 求第30位数字是多少,用递规和非递归两种方法算法实现. ...

  8. keepalive + nginx 搭建高可用集群动态网站

    环境准备: 两台节点部署keepalived,并且设为互为主从,实现高可用. 两台从节点部署nginx以及相关组件,作为真实服务器实现动态网站上线. 一.MASTER(BACKUP)节点下载keepa ...

  9. laradock 部署 php 环境 和 laravel/lumen 框架

    环境是windows 10 版本1809,docker 版本18.09.0 首先是下载docker.git, 具体可以参考 http://laradock.io/ 要求 Docker >= 17 ...

  10. 解决react项目中跨域和axios封装使用

    最新几天学了一下react,发现了几个问题,估计新入坑的同学们也会遇到,下面我先列出来几点 1.请求跨域问题 2.如何发起请求 3.axios的简单封装 全局安装create-react-app脚手架 ...