2012-07-18 65 views
2

我有一个托管C++类,其所有成员变量在构造函数中初始化。感兴趣的成员是一个数组。我从C#项目的.cs文件调用它,并将这两个项目与第一个项目的dll链接起来。但是,函数说明一个或多个参数不正确,因此无法成功使用。托管C++和C#

类声明和函数声明如下。两者都在.h文件中。现在

,我想调用该函数在.cs文件如下:

var Driver = new Driver(); 
long status = Driver.Config2("CAN0", 8, Driver.AttrIdList, Driver.AttrValueList); 
Console.WriteLine(status); 

如果功能配置正确执行,应该输出0。但是,我得到一个负数和在查询供应商提供的表时,它指出一个或多个参数设置不正确。我不知道如何克服这一点,因为我是托管C++的新手。所有的帮助将不胜感激。 谢谢。

代码声明如下:

public ref class Driver 
    { 
    public: 

     NCTYPE_STATUS Status; 
     NCTYPE_OBJH TxHandle; 
     MY_NCTYPE_CAN_FRAME Transmit; 
     array<NCTYPE_UINT32>^ AttrIdList; 
     array<NCTYPE_UINT32>^ AttrValueList; 
     array<char>^ Data; 
     NCTYPE_UINT32 Baudrate; 

    public: 
     Driver() 
     { 
      Baudrate = 1000000; 
      TxHandle = 0; 
      AttrIdList = gcnew array<NCTYPE_UINT32>(8); 
      AttrValueList = gcnew array<NCTYPE_UINT32>(8); 
      AttrIdList[0] =   NC_ATTR_BAUD_RATE; 
      AttrValueList[0] =  Baudrate; 
      AttrIdList[1] =   NC_ATTR_START_ON_OPEN; 
      AttrValueList[1] =  NC_TRUE; 
      AttrIdList[2] =   NC_ATTR_READ_Q_LEN; 
      AttrValueList[2] =  0; 
      AttrIdList[3] =   NC_ATTR_WRITE_Q_LEN; 
      AttrValueList[3] =  1; 
      AttrIdList[4] =   NC_ATTR_CAN_COMP_STD; 
      AttrValueList[4] =  0; 
      AttrIdList[5] =   NC_ATTR_CAN_MASK_STD; 
      AttrValueList[5] =  NC_CAN_MASK_STD_DONTCARE; 
      AttrIdList[6] =   NC_ATTR_CAN_COMP_XTD; 
      AttrValueList[6] =  0; 
      AttrIdList[7] =   NC_ATTR_CAN_MASK_XTD; 
      AttrValueList[7] =  NC_CAN_MASK_XTD_DONTCARE; 

      interior_ptr<NCTYPE_UINT32> pstart (&AttrIdList[0]); 
      interior_ptr<NCTYPE_UINT32> pend (&AttrIdList[7]); 


      Data = gcnew array<char>(8); 
      for (int i=0; i<8;i++) 
       Data[i]=i*2; 

     } 

我也有另一种方法正下方被声明为Config功能如下:

NCTYPE_STATUS Config2 (String^ objName, int numAttrs, array<unsigned long>^ AttrIdList, array<unsigned long>^ AttrValueList) 
    { 
     msclr::interop::marshal_context^ context = gcnew msclr::interop::marshal_context(); 
     const char* name = context->marshal_as<const char*>(objName); 


     char* name_unconst = const_cast<char*>(name); 

     return ncConfig (name_unconst, 8, nullptr, nullptr); 
     delete context; 

    } 

程序编译和构建,这是一个运行时错误。我猜测它与在函数Config2中传递的两个nullptr有关,但是如果我用参数AttrIdList和AttrValueList替换它们,编译器会提供一个错误: 无法将参数3从'cli :: array ^'转换为'NCTYPE_ATTRID_P'

BTW:NCTYPE_STATUS为无符号长整型,而NCTYPE_ATTRID_P为无符号长整型*。

+0

请提供所有适用的代码,例如什么是类型NCTYPE_STRING和NCTYPE_UINT32 – 2012-07-18 20:35:41

+0

NCTYPE_UINT32是无符号长整型,而NCTYPE_STRING是char *。所有这些都是在供应商的库中声明的,并且包含太长的时间。 – Nathan822 2012-07-18 20:39:49

+0

CLR中的无符号长整型为32位 – Nathan822 2012-07-18 20:42:58

回答

0

cannot convert parameter 3 from 'cli::array^' to 'NCTYPE_ATTRID_P'
NCTYPE_ATTRID_P is unsigned long*

可以托管数组不传递到一个纯粹的本地C++的功能,首先需要“转换”到一个固定的无符号长*指针。

这里有一个办法做到这一点:

unsigned long* ManagedArrayToFixedPtr(array<unsigned long>^input) 
{ 
    pin_ptr<unsigned long> pinned = &input[0]; 
    unsigned long* bufferPtr = pinned; 

    unsigned long* output = new unsigned long[input->Length]; 
    memcpy_s(output, input->Length, bufferPtr, input->Length); 

    return output; 
} 

测试功能:

array<unsigned long>^ larray = gcnew array<unsigned long> {2,4,6,8,10,12,14,16}; 
unsigned long* lptr = ManagedArrayToFixedPtr(larray); //returns pointer to 2 

编辑:
Rememer到#include "windows.h"能够使用memcpy_s功能!