可以让Python向C#发送一个可变长度的字符串数组吗?

时间:2018-02-11 14:02:54

标签: c# python dll ctypes unmanagedexports

这是this question的反向/补充。

我正在使用Unmanaged Exportsctypes,并希望将可变长度的字符串列表从Python传递给C#,然后返回一个字符串。

我尝试了四种C#+ Python变体 - 没有成功。任何人都可以完成工作吗?

C#

    /* no ref, convert to list */
    [DllExport("combineWords1", CallingConvention = CallingConvention.Cdecl)]
    [return: MarshalAs(UnmanagedType.AnsiBStr)]
    public static string CombineWords1(object obj)
    {
        var l = (List<string>) obj;
        return string.Join(",", l.ToArray());
    }

    /* no ref, convert to array */
    [DllExport("combineWords2", CallingConvention = CallingConvention.Cdecl)]
    [return: MarshalAs(UnmanagedType.AnsiBStr)]
    public static string CombineWords2(object obj)
    {
        var l = (string[]) obj;
        return string.Join(",", l);
    }

    /* ref, convert to list */
    [DllExport("combineWords3", CallingConvention = CallingConvention.Cdecl)]
    public static void CombineWords3(ref object obj)
    {
        var l = (List<string>)obj;
        obj = string.Join(",", l.ToArray());
    }

    /* ref, convert to array */
    [DllExport("combineWords4", CallingConvention = CallingConvention.Cdecl)]
    public static void CombineWords4(ref object obj)
    {
        var l = (string[]) obj;
        obj = string.Join(",", l.ToArray());
    }

的Python

import ctypes
from comtypes.automation import VARIANT

dll = ctypes.cdll.LoadLibrary("<...>")
l = ["Hello", "world"]

# 1 
dll.combineWords1.argtypes = [ctypes.POINTER(VARIANT)]
dll.combineWords1.restype = ctypes.c_char_p
obj = VARIANT(l)
dll.combineWords1(obj)
#WindowsError: [Error -532462766] Windows Error 0xE0434352

# 2
dll.combineWords2.argtypes = [ctypes.POINTER(VARIANT)]
dll.combineWords2.restype = ctypes.c_char_p
obj = VARIANT(l)
dll.combineWords2(obj)
# WindowsError: [Error -532462766] Windows Error 0xE0434352

# 3
dll.combineWords3.argtypes = [ctypes.POINTER(VARIANT)]
obj = VARIANT(l)
dll.combineWords3(obj)
#WindowsError: [Error -532462766] Windows Error 0xE0434352

# 4
dll.combineWords4.argtypes = [ctypes.POINTER(VARIANT)]
obj = VARIANT(l)
dll.combineWords4(obj)
#WindowsError: [Error -532462766] Windows Error 0xE0434352      

1 个答案:

答案 0 :(得分:4)

您可以像这样声明C#:

[DllExport("combinewords", CallingConvention = CallingConvention.Cdecl)]
[return: MarshalAs(UnmanagedType.LPWStr)] // matches python's c_wchar_p
public static string CombineWords(object obj) // object matches python's VARIANT
{
    var array = (object[])obj; // if obj is an array, it will always be an array of object
    return string.Join(", ", array);
}

这样的python方面:

import ctypes
from ctypes import *
from comtypes.automation import VARIANT

dll = ctypes.cdll.LoadLibrary("exported") # the dll's name
dll.combinewords.argtypes = [VARIANT] # python VARIANT = C# object
dll.combinewords.restype = ctypes.c_wchar_p

# create a VARIANT from a python array
v = VARIANT(["hello", "world"])
print(dll.combinewords(v))  

注意:我使用的unicode声明更好。安西已成为过去。