Возврат вектора из нативной dll C ++ в управляемую dll C ++ / CLI - PullRequest
1 голос
/ 20 мая 2011

Я пишу оболочку CLI / C ++ вокруг собственной библиотеки C ++, которую нельзя изменить. Одна из функций нативной DLL возвращает вектор неуправляемых объектов. Что было бы лучшим способом обернуть этот вектор в моей оболочке CLI? Оболочка CLI будет использоваться приложением C #.

class __declspec(dllexport) Instrument
{
  public:
        Instrument();
        ~Instrument();
        string _type;
        unsigned int _depth;
}

В нативной DLL есть функция getInstruments (), которую я пытаюсь обернуть

 class __declspec(dllexport) InstrumentList
   {
       InstrumentList();
       ~InstrumentList();
       vector<Instrument*> getInstruments();
   }

Поэтому мне нужно обернуть класс инструмента в управляемом классе и обернуть класс InstrumentList в управляемом классе. У меня есть класс Instrument, но мне нужно преобразовать вектор, возвращаемый getInstruments (), во что-то эквивалентное, которое может вернуть оболочка CLI для InstrumentList.

Ответы [ 2 ]

4 голосов
/ 20 мая 2011

Если вы не хотите отложить маршалинг Instrument::_type до тех пор, пока к его управляемому фасаду не будет получен доступ, вам следует начать:

public ref class InstrumentM
{
    String^ _type;
    unsigned _depth;

internal:
    explicit InstrumentM(Instrument const& i)
      : _type(gcnew String(i._type.c_str())),
        _depth(i._depth)
    { }

public:
    property String^ Type { String^ get() { return _type; } }
    property unsigned Depth { unsigned get() { return _depth; } }
};

public ref class InstrumentListM
{
    InstrumentList* _list;

public:
    InstrumentListM() : _list(new InstrumentList()) { }
    ~InstrumentListM() { this->!InstrumentListM(); }
    !InstrumentListM()
    {
        delete _list;
        _list = nullptr;
    }

    array<InstrumentM^>^ GetInstruments()
    {
        if (!_list)
            throw gcnew ObjectDisposedException(L"_list");

        vector<Instrument*> const& v = _list->getInstruments();
        array<InstrumentM^>^ ret = gcnew array<InstrumentM^>(v.size());
        for (int i = 0, i_max = ret->Length; i != i_max; ++i)
            if (v[i])
                ret[i] = gcnew InstrumentM(*v[i])
        return ret;
    }
};
4 голосов
/ 20 мая 2011

Возможно, вы вообще не захотите оборачивать InstrumentList.

Просто используйте одну из стандартных коллекций .NET (к которой вы можете получить доступ из C ++ / CLI) и создайте коллекцию ваших оболочек Instrument.Я использую ObservableCollection с тех пор, как хочу привязать данные к своим коллекциям.

Пример:

public ref class MyManagedType
{
    public:
        MyManagedType(MyNativeType* pObject) { m_pObject = pObject };

    private:
        MyNativeType* m_pObject;
}

Затем создайте управляемую коллекцию следующим образом:

ObservableCollection<MyManagedType^>^ managedCollection = gcnew ObservableCollection<MyManagedType^>();

Наконецдобавьте объекты в коллекцию:

managedCollection->Add(gcnew MyManagedType(pNativeObject));

Немного усилий для синхронизации собственных и управляемых коллекций, но это хорошо работает.

...