Cómo obtener el contenido de la matriz de C ++ dll en C #
Quiero usar funciones de DLL en C ++ con C #.
Almaceno datos de cadena en un vector.
Mi archivo C ++ contiene:
#include "stdafx.h"
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
extern "C" __declspec(dllexport) std::vector<std::string> GetProduct();
std::vector<std::string> GetProduct()
{
  std::vector<std::string> vectProduct;
  vectProduct.push_back("Citroen");
  vectProduct.push_back("C5");
  vectProduct.push_back("MOP-C5");
  return vectProduct;
}
C ª#
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.IO;
using System.Runtime.InteropServices;
namespace ConsoleApplication
{
    class Program
    {
        [DllImport("ProductLibrary.dll", CallingConvention = CallingConvention.Cdecl)]
        public static extern StringBuilder GetProduct();
        static void Main(string[] args)
        {
            StringBuilder vectProduct_impl = GetProduct();
        }
    }
}
No sé cómo seguir navegando por la matriz en c #. No sé si el uso de vector es óptimo. si tienes otra solución, estoy listo.
Por favor ayuda.
Respuestas
Mi forma favorita de pasar una matriz de cadenas C ++ -> C # es mediante el uso de un delegado.
C#:
// If possible use UnmanagedType.LPUTF8Str
// or under Windows rewrite everything to use 
// wchar_t, std::wstring and UnmanagedType.LPWStr
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate void AddAnsi([MarshalAs(UnmanagedType.LPStr)] string str);
[DllImport("CPlusPlusSide.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void TestReturnArrayStrings(AddAnsi add);
y entonces
var lst = new List<string>();
TestReturnArrayStrings(lst.Add);
foreach (string str in lst)
{
    Console.WriteLine(str);
}
Y C ++:
#include <string>
#include <vector>
extern "C"
{
    __declspec(dllexport) void TestReturnArrayStrings(void (add)(const char* pstr))
    {
        std::string str1 = "Hello";
        std::string str2 = "World";
        add(str1.data());
        add(str2.data());
        // Example with std::vector
        add("--separator--"); // You can even use C strings
        std::vector<std::string> v = { "Foo", "Bar" };
        // for (std::vector<std::string>::iterator it = v.begin(); it != v.end(); ++it)
        for (std::vector<std::string>::const_iterator it = v.begin(); it != v.end(); ++it)
        {
            add(it->data());
        }
        add("--separator--"); // You can even use C strings
        // With C++ 11
        // for (auto& it: v)
        for (const auto& it: v)
        {
            add(it.data());
        }
    }
}
Aquí el "truco" es que C # pasa a C ++ un delegado al List<string>.Add()método, y C ++ "llena" directamente el C # List<>. La memoria administrada por C ++ permanece en el lado C ++, la memoria administrada por C # permanece en el lado C #. Sin problemas de propiedad de memoria cruzada. Como puede imaginar, es bastante fácil expandir el "truco" a cualquier otro .Add()método, como HashSet<string>, o Dictionary<string, string>.
Como nota al margen, he creado un github con muchos ejemplos sobre la clasificación entre C / C ++ y C # (tanto .NET Framework como .NET Core / 5.0).
Una forma de hacerlo es utilizar la estructura SAFEARRAY de COM, ya que es compatible con .NET (el asignador de .NET utilizado por P / Invoke es el asignador de COM), incluida la mayoría de los subtipos asociados, como BSTR .
Entonces, en C / C ++, puede definir esto:
extern "C" __declspec(dllexport) LPSAFEARRAY GetProduct();
LPSAFEARRAY GetProduct()
{
    LPSAFEARRAY psa = SafeArrayCreateVector(VT_BSTR, 0, 3);
    LONG index = 0;
    // _bstr_t is a smart class that frees allocated memory automatically
    // it needs #include <comdef.h>
    // but you can also use raw methods like SysAllocString / SysFreeString
    _bstr_t s0(L"Citroen"); // could use "Citroen" if you really want ANSI strings
    // note SafeArrayPutElement does a copy internally
    SafeArrayPutElement(psa, &index, s0.GetBSTR());
    index++;
    _bstr_t s1(L"C5");
    SafeArrayPutElement(psa, &index, s1.GetBSTR());
    index++;
    _bstr_t s2(L"MOP - C5");
    SafeArrayPutElement(psa, &index, s2.GetBSTR());
    index++;
    return psa;
}
Y en C #, puede definir esto:
[DllImport("ProductLibrary.dll", CharSet = CharSet.Unicode)]
[return: MarshalAs(UnmanagedType.SafeArray)]
public static extern string[] GetProduct();