首页 > 解决方案 > 如何在 ctypes 中接收来自 .dll 的回调函数消息?

问题描述

我正在尝试使用 Ctypes 调用 .dll 函数。我有一个问题是获取回调函数值。如果我调用回调函数,该函数不会给出值。我的python代码有什么问题?

void __stdcall callbackmessage(int msgT, char* msg, void* data)
typedef void (__stdcall *CALLBACK)(int, char*, void*);
extern "C" _declspec(dllimport) void    __stdcall callbackfunc(CALLBACK callback, void* userdata);
import sys, ctypes
from ctypes import *
from ctypes import wintypes, POINTER

Load_DLL = WinDLL("test.dll")
sys.stdout = open('stdout.txt', 'w')

CONNECT= 4116
DISCONNECTED = 4113

def callbackmessage(msgT, msg, data):
     if msgT == 4113:
         print (b"disconnected!")
     elif msgT == 4116:
         print ("Connect!!")


CALLBACK = WINFUNCTYPE( c_int, POINTER(c_char), POINTER(c_void_p))

callbackfunc = Load_DLL.callbackfunc
callbackfunc.argtypes = [CALLBACK, c_void_p]
callbackfunc.restype = c_void_p
CALLBACKMESSAGE = CONNECT|DISCONNECTED
messageprocess = CALLBACK(callbackmessage)

callbackfunc(messageprocess, CALLBACKMESSAGE)

标签: pythonctypes

解决方案


提到 [Python 3.Docs ]:ctypes - Python 的外部函数库

我看到代码有几个问题:

  1. callbackmessagePython签名不反映真实的签名(来自C)。要更正它,请将其更改为:

    CALLBACK = WINFUNCTYPE(None, c_int, POINTER(c_char), c_void_p)  # Adding 1st arg which is the return type; 3rd arg might also be c_char_p
    
  2. 我不确定callbackfunc如何解释其第二个参数,但将CALLBACKMESSAGE放在那里似乎是错误的。我会这样做:

    callbackfunc(messageprocess, byref(c_int(CALLBACKMESSAGE)))
    
  3. 你确定(在C中)callbackfunc_declspec(dllimport)吗?我认为应该是__declspec(dllexport)

  4. callbackfunc.restype = None(次要的)。voidvoid指针 ( c_void_p )不是一回事

  5. 最后但同样重要的是:这是一个逻辑错误:CALLBACKMESSAGE = CONNECT | DISCONNECTED这将产生一个值4117,如果这将是callbackmessage的第一个参数它不会打印任何东西。所以,选择一个值(例如CALLBACKMESSAGE = CONNECT


推荐阅读