1

我有char**一个 C 结构,它在 C 代码中分配为Nx128矩阵。在 C# 中,我有一个字符串数组,我想将此数组复制到 char 双指针,而不重新分配任何内容。我试过这个:

public void StringArrayToPtr(IntPtr ptr, string[] array) 
{

    for (int i = 0; i < array.Length; i++)
    {
        char[] chars = (array[i] + '\0').ToCharArray();

        Marshal.Copy(chars, 0, IntPtr.Add(ptr, 128*i), chars.Length);
    }
}

但这不起作用。有人知道如何进行这样的复制吗?

更新:

以下是 mychar** names在 C 代码中分配 3 个项目的方式:names = (char **) MallocArray2D (3, 128, sizeof ( char ));

以下是该MallocArray2D方法的详细信息:

void ** MallocArray2D (
     int  n1,
     int  n2,
     int  size_elem )
{
    void ** p2;
    void * p1;
    size_t i;

    p1 = (void *) malloc (size_elem * n1 * n2); 

    p2 = (void **) malloc (n1 * sizeof ( void * ));


    for ( i = 0 ; i < n1 ; i++ )
    {
        p2[i] = (char *) p1 + size_elem * n2 * i;
    }

    return p2;
}

MallocArray2D方法被调用到MallocImage我的 C# 代码中公开的 a 中。

这是MallocImageC 代码中有趣的方法部分:

int MallocImage (
     IMAGE *  image,
     int           nxyz,
     int           nvar )
{
    //... Allocating others objects

    image->names = (char **) MPDSMallocArray2D ( nvar, 128, sizeof ( char ));

}

现在我的 C# 公开MallocImage方法:

[DllImport(DLL_PATH, CallingConvention = CallingConvention.Cdecl)]
public static extern int MallocImage([In, Out]Image image, int nXYZ, int nVar);

// My Image class
[StructLayout(LayoutKind.Sequential)]
class Image {

    private IntPtr names;

    public string[] Names {

        set {ArrayOfStringToPtr(names, value);}

    }

    // Constructor
    public Image(int nVar, int nXYZ) {

        MallocImage(this, nXYZ, nVar);

    }

}


// Somewhere else in my code
image.Names = new string[] {"porosity", "duplicity", "facies"];
4

1 回答 1

1

ASystem.Char是一个 unicode 16 位字符 [ MSDN ]。您可能使用 ASCII 字符串。

这里的指针算术似乎是错误的,因为您正在使用指向三个指针数组的指针,您可能需要使用 : 获取字符串的地址,Marshal.ReadIntPtr(ptr, i * IntPtr.Size)因此生成的代码将是:

public static void StringArrayToPtr(IntPtr ptr, string[] array)
{
    for (int i = 0; i < array.Length; i++)
    {
        byte[] chars = System.Text.Encoding.ASCII.GetBytes(array[i] + '\0');
        Marshal.Copy(chars, 0, Marshal.ReadIntPtr(ptr, i * IntPtr.Size), chars.Length);
    }
}
于 2013-06-13T16:09:42.740 回答