在WinCE

时间:2017-01-19 17:19:40

标签: c# c++ arraylist compact-framework marshalling

我试图在Windows CE程序和紧凑框架2.0上将以下结构从c ++转换为c#。我在编组字符串方面遇到了很多困难。

我有这个c ++代码:

#define Console_Parameters_MAX 50

struct AllParameters {
  Parameter Parameters[Console_Parameters_MAX];
} ;

struct Parameter { 
  int Index;
  int Id; 
  char Value[20];
};

extern "C" {
   BOOL GetAllConsoleParameters(AllParameters *pItem);
}

这是相应的c#代码:

[StructLayout(LayoutKind.Sequential)]
public struct AllParameters {
  [MarshalAs(UnmanagedType.ByValArray, SizeConst = 50)]
  public Parameter[] Parameters
}

[StructLayout(LayoutKind.Sequential)]
public struct Parameter {
  public int Index;
  public int Id;
  [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 20)]
  public byte[] Value;
}

[DllImport("exemple.dll", SetLastError = true)]
public static extern bool GetAllConsoleParameters([MarshalAs(UnmanagedType.Struct)] ref AllParameters pItem);

这就是我调用它的方式:

AllParameters item = new AllParameters();
if (AppAPI.GetAllConsoleParameters(ref item)) {
   var array = item.Parameters;
}

当我调用GetAllConsoleParameters时,我得到异常NotSupportedException。我尝试了很多配置但没有成功。

任何人都可以建议如何实现它吗?

提前致谢

3 个答案:

答案 0 :(得分:0)

这适用于Windows桌面。您可能必须在C DLL和C#DllImport属性中将调用约定更改为Cdecl,因为我在此处读到Cdecl是Windows CE上的标准:https://msdn.microsoft.com/en-us/library/system.runtime.interopservices.callingconvention(v=vs.110).aspx

C代码:

extern "C" {
  __declspec(dllexport) BOOL __stdcall GetAllConsoleParameters(AllParameters *pItem)
  {
    pItem->Parameters[0].Index = 0;
    pItem->Parameters[0].Id = 42;
    CopyMemory(&pItem->Parameters[0].Value[0], "Hello World", 12);
    pItem->Parameters[1].Index = 1;
    pItem->Parameters[1].Id = 43;
    CopyMemory(&pItem->Parameters[1].Value[0], "Hello World 43", 15);
    return TRUE;
  }
}

C#代码:

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
struct Parameter
{
    int Index;
    int Id;
    //char Value[20];
    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 20)]
    string Value;
};

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
struct AllParameters
{
    //Parameter Parameters[Console_Parameters_MAX];
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 50)]
    Parameter[] Parameters;
};

class Program
{
    [DllImport("MarshalC.dll", CallingConvention = CallingConvention.StdCall)]
    [return: MarshalAs(UnmanagedType.Bool)]
    static extern bool GetAllConsoleParameters(ref AllParameters pItem);

    static void Main(string[] args)
    {
        var size = Marshal.SizeOf<AllParameters>();
        AllParameters all = new AllParameters();
        bool result = GetAllConsoleParameters(ref all);
    }
}

答案 1 :(得分:0)

我会这样做

        [StructLayout(LayoutKind.Sequential)]
        public struct AllParameters {
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 50)]
            public Parameter[] Parameters;
        }

        [StructLayout(LayoutKind.Sequential)]
        public struct Parameter {
          public int Index;
          public int Id;
          [MarshalAs(UnmanagedType.ByValArray, SizeConst = 20)]
          public byte[] Value;
        }

        [DllImport("exemple.dll", SetLastError = true, CallingConvention = CallingConvention.Cdecl)]
        public static extern bool GetAllConsoleParameters(ref IntPtr pItem);


        static void Main(string[] args)
        {
            AllParameters allParameters = new AllParameters();
            allParameters.Parameters = new Parameter[50];
            IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(allParameters));

            int z = Marshal.SizeOf(allParameters);

            if (GetAllConsoleParameters(ref ptr))
            {
                Marshal.PtrToStructure(ptr, allParameters);
                Parameter[] parameters = allParameters.Parameters;
            }

        }

答案 2 :(得分:0)

根据我的解决方案,c ++代码:

/* - not used
#define Console_Parameters_MAX 50

struct AllParameters {
  Parameter Parameters[Console_Parameters_MAX];
} ;
*/

struct Parameter { 
  int Index;
  int Id; 
  char Value[20];
};

extern "C" {
   BOOL GetAllConsoleParameters(Parameter pItem[], int size);
}

和相应的c#代码:

/* - not used
[StructLayout(LayoutKind.Sequential)]
public struct AllParameters {
  [MarshalAs(UnmanagedType.ByValArray, SizeConst = 50)]
  public Parameter[] Parameters
}
*/

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
public struct Parameter {
  public int Index;
  public int Id;
  [MarshalAs(UnmanagedType.ByValArray, SizeConst = 20)]
  public byte[] Value;
}

[DllImport("exemple.dll", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern bool GetAllConsoleParameters([MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1), Out] ConsoleParameter[] myStruct, int size);

和代码调用:

ConsoleParameter[] item = new ConsoleParameter[50];
if (AppAPI.GetAllConsoleParameters(item, 50)) {
   var array = item;
}

非常感谢你的帮助