17

所以我有下一个 C++ 代码:

#ifdef WIN32
#  undef CALLBACK
#  define CALLBACK __stdcall
#else
#  define CALLBACK
#endif


#include <iostream>
#include <vector>

namespace OdeProxy {

    typedef std::vector< double > state_type;
    typedef void (CALLBACK *System)( const state_type &, state_type &, const double);
    typedef void (CALLBACK *Observer)( const state_type &, double);

    class Ode {
    public:
        state_type initialConditions;
        System system;
        Observer observer;
        double from;
        double to;
        double step;
    };
}

和 .i 文件:

/* File : MyProject.i */
%module MyProject

%{
#include "C++/OdeProxy.h"
%}

%include "std_vector.i"
%include "C++/OdeProxy.h"

%template(state_type) std::vector<double>;


//// Delegate realated stuff ////
%typemap(cstype) void (*)( const state_type &, state_type &, const double) "SystemDelegate";
%typemap(imtype) void (*)( const state_type &, state_type &, const double) "SystemDelegate";

%typemap(cstype) void (*)( const state_type &, double) "ObserverDelegate";
%typemap(imtype) void (*)( const state_type &, double) "ObserverDelegate";

我受到这个线程的启发而创作。生成代码。

但是我不明白如何获取代码

using OdeLibrary;

namespace OdeTest
{
    class Program
    {
        static void Main(string[] args)
        {
            //var lam = new OdeLibrary.SWIGTYPE_p_f_r_q_const__std__vector__double___double__void()
            var ode = new Ode{
                from = 0,
                to = 10,
                initialConditions = new state_type(new[]{1,2,3}),
                step = 0.01,
                observer = (x, dxdt, t) => { return; }
           };
        }
    }
}

编译。错误:

Error   Cannot convert lambda expression to type 'OdeLibrary.SWIGTYPE_p_f_r_q_const__std__vector__double___double__void' because it is not a delegate type

哪里SWIGTYPE_p_f_r_q_const__std__vector__double___double__void看起来像这样:

/* ----------------------------------------------------------------------------
 * This file was automatically generated by SWIG (http://www.swig.org).
 * Version 2.0.9
 *
 * Do not make changes to this file unless you know what you are doing--modify
 * the SWIG interface file instead.
 * ----------------------------------------------------------------------------- */

namespace OdeLibrary {

using System;
using System.Runtime.InteropServices;

public class SWIGTYPE_p_f_r_q_const__std__vector__double___double__void {
  private HandleRef swigCPtr;

  internal SWIGTYPE_p_f_r_q_const__std__vector__double___double__void(IntPtr cPtr, bool futureUse) {
    swigCPtr = new HandleRef(this, cPtr);
  }

  protected SWIGTYPE_p_f_r_q_const__std__vector__double___double__void() {
    swigCPtr = new HandleRef(null, IntPtr.Zero);
  }

  internal static HandleRef getCPtr(SWIGTYPE_p_f_r_q_const__std__vector__double___double__void obj) {
    return (obj == null) ? new HandleRef(null, IntPtr.Zero) : obj.swigCPtr;
  }
}

}

所以我想知道应该在.i文件中更改什么或添加到 c# 生成的包装器中以获得将我的 C# lambda 作为委托传递给 C++ 类的能力?

4

1 回答 1

3

得到这个工作以下配置:

public class StateTypeCustomMarshaller : ICustomMarshaler
{
    public static ICustomMarshaler GetInstance(string s)
    {
        return new StateTypeCustomMarshaller();
    }

    public object MarshalNativeToManaged(IntPtr pNativeData)
    {
        return new state_type(pNativeData, false);
    }

    public IntPtr MarshalManagedToNative(object ManagedObj)
    {
        throw new NotImplementedException();
    }

    public void CleanUpNativeData(IntPtr pNativeData)
    {
        throw new NotImplementedException();
    }

    public void CleanUpManagedData(object ManagedObj)
    {
    }

    public int GetNativeDataSize()
    {
        throw new NotImplementedException();
    }
}

public delegate void ObserverDelegate(
        [MarshalAs(UnmanagedType.CustomMarshaler, MarshalTypeRef = typeof(StateTypeCustomMarshaller))]state_type state,
        double d);

对应的 .i 文件是:

/* File : MyProject.i */
%module MyProject

%include "std_vector.i"
%template(state_type) std::vector<double>;

//// Delegate realated stuff ////
%typemap(csin) void (*)(OdeProxy::state_type&, double) "$csinput";
%typemap(cstype) void (*)(OdeProxy::state_type&,double) "ConsoleApplication2.Helpers.ObserverDelegate";
%typemap(imtype) void (*)(OdeProxy::state_type&, double) "ConsoleApplication2.Helpers.ObserverDelegate";
%typemap(csvarout) void (*)(OdeProxy::state_type&, double) %{
get {
  return $imcall;
} %}

%{
    #include "OdeProxy.h"
%}

%include "OdeProxy.h"

注意:我已经尝试过对状态类型的非常量引用,但也可以使用常量引用。

于 2013-05-29T23:51:03.030 回答