如何将字符串数组复制到非托管char双指针?

时间:2013-06-13 15:55:22

标签: c# char marshalling unmanaged-memory

我在C结构中有一个char**,它在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);
    }
}

但这不起作用。 有人知道如何执行此类副本吗?

更新:

以下是我的char** names在3个项目的C代码中的分配方式: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#代码中公开。

以下是C代码中的MallocImage方法有趣部分:

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"];

1 个答案:

答案 0 :(得分:1)

System.Char是一个16位字符的unicode [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);
    }
}