我有一个调用Windows dll(fbwflib.dll)的C#应用程序。 从该dll调用的函数(FbwfIsFilterEnabled)需要通过引用传递两个参数。
在C#中,我的代码是:
public class FBWF_Utilities
{
[DllImport("fbwflib.dll", SetLastError = true)]
static extern UIntPtr FbwfIsFilterEnabled(
[MarshalAs(UnmanagedType.U1)]
ref bool currentSession,
[MarshalAs(UnmanagedType.U1)]
ref bool nextSession
);
[DllImport("fbwflib.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.U4)]
static extern uint FbwfEnableFilter();
[DllImport("fbwflib.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.U4)]
static extern uint FbwfDisableFilter();
public static bool IsEnabledCurrent()
{
bool xCurrent = false;
bool xNext = false;
if (IsInstalled())
{
try
{
FbwfIsFilterEnabled(ref xCurrent, ref xNext);
}
catch (Exception ex)
{
MessageBox.Show("FBWF IsEnabledCurrent exception:\n\n" + ex.Message);
return false;
}
}
return xCurrent;
}
现在我需要将此代码转换为Python代码。
我尝试过:
@staticmethod
def IsEnabled():
dllObject = ctypes.cdll.LoadLibrary(r'C:\Windows\System32\fbwflib.dll')
dllObject.FbwfIsFilterEnabled.argtypes = [ctypes.c_int, ctypes.c_int]
dllObject.FbwfIsFilterEnabled.restype = None
current = ctypes.c_int(0)
next = ctypes.c_int(0)
dllObject.FbwfIsFilterEnabled(ctypes.byref(current, 0), ctypes.byref(next, 0))
messagebox.showinfo("", "Current: " + current + "\n" + "Next: " + next)
但是我有一个错误: ctypes.ArgumentError:参数1:错误类型
我尝试了其他参数化,但是没有任何好的结果。
通过引用加载带有参数的库的正确方法是什么?
其他不带参数的功能效果很好
@staticmethod
def EnableFilter():
dllObject = ctypes.cdll.LoadLibrary(r'C:\Windows\System32\fbwflib.dll')
dllObject.FbwfEnableFilter()
@staticmethod
def DisableFilter():
dllObject = ctypes.cdll.LoadLibrary(r'C:\Windows\System32\fbwflib.dll')
dllObject.FbwfDisableFilter()
工作正常,没有任何错误。
给我带来麻烦的函数的简短描述在这里: https://www.pinvoke.net/default.aspx/fbwflib.FbwfIsFilterEnabled
编辑:
根据卢克的建议更改ctypes.pointer(谢谢),现在我遇到了另一个错误:
程序调用时参数不足(缺少8个字节)或调用约定错误
参考行:
dllObject.FbwfIsFilterEnabled(ctypes.byref(current, 0), ctypes.byref(next, 0))
将行更改为:
dllObject.FbwfIsFilterEnabled(ctypes.POINTER(current), ctypes.POINTER(next))
我收到错误: 类型错误:必须为ctypes类型
让我知道byref调用可能是正确的方法。
根据我的C#代码更改考虑的布尔类型:
@staticmethod
def IsEnabled():
dllObject = ctypes.cdll.LoadLibrary(r'C:\Windows\System32\fbwflib.dll')
dllObject.FbwfIsFilterEnabled.argtypes = [ctypes.POINTER(ctypes.c_bool), ctypes.POINTER(ctypes.c_bool)]
dllObject.FbwfIsFilterEnabled.restype = None
current = ctypes.c_bool()
next = ctypes.c_bool()
dllObject.FbwfIsFilterEnabled(ctypes.byref(current), ctypes.byref(next))
messagebox.showinfo("", "Current: " + current + "\n" + "Next: " + next)
我收到相同的错误: 没有足够的参数(缺少8个字节)或错误的调用约定调用了过程
我认为解决方案就在附近,但它似乎缺少一些细节
解决方案:
解决方案终于从
开始ctypes.windll.LoadLibrary
根据卢克的建议。 我现在已经删除了argtypes和restype,并且在接收到可以为0或1 a的c_long值后已经获得了布尔条件。
简单的代码(我现在可以这样说)是:
@staticmethod
def IsEnabled():
dllObject = ctypes.windll.LoadLibrary(r'C:\Windows\System32\fbwflib.dll')
c = ctypes.c_int()
# dllObject.FbwfIsFilterEnabled.argtypes = [ctypes.pointer(ctypes.c_int), ctypes.pointer(ctypes.c_int)]
# dllObject.FbwfIsFilterEnabled.restype = ctypes.c_void_p
current = ctypes.c_int()
next = ctypes.c_int()
dllObject.FbwfIsFilterEnabled(ctypes.byref(current), ctypes.byref(next))
messagebox.showinfo("", "Current: " + str(bool(current)) + "\n" + "Next: " + str(bool(next)))
我希望它对其他人有用,以免像我这样在初始代码附近的解决方案上花费这么多时间,但由于此dll的“性质”,它需要花费几个小时才能解决嵌入式系统以及该特定库的安装。
非常感谢
答案 0 :(得分:0)
尝试替换行
dllObject.FbwfIsFilterEnabled.argtypes = [ctypes.c_int, ctypes.c_int]
使用
dllObject.FbwfIsFilterEnabled.argtypes = [POINTER(ctypes.c_int), POINTER(ctypes.c_int)]
我还注意到,该函数已定义为返回UIntPtr
,因此您可能还想尝试替换该行
dllObject.FbwfIsFilterEnabled.restype = None
使用
dllObject.FbwfIsFilterEnabled.restype = c_void_p
但是,我没有该DLL的副本,因此无法测试该更改是否有效。