Python Load Windows库C#转换

时间:2018-09-20 11:49:57

标签: python ctypes byref

我有一个调用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的“性质”,它需要花费几个小时才能解决嵌入式系统以及该特定库的安装。

非常感谢

1 个答案:

答案 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的副本,因此无法测试该更改是否有效。