5

C からジェネリック メソッド (静的および非静的の両方) を呼び出す方法がわかりません。

最大の原因は、ジェネリック メソッドの呼び出しが本質的に文書化されていないことです。mono リポジトリには例がありませんが、ドキュメントではこれが可能であると言及されています。

If you want to invoke generic methods, you must call the method on the "inflated" class, which you can obtain from the mono_object_get_class()


MonoClass *clazz;
MonoMethod *method;

clazz = mono_object_get_class (obj);

/*
 * If there are more Add methods declared, you
 * may use mono_method_desc_search_in_class (clazz, ":Add(T)"),
 * you must substitute ":Add(T)" with the correct type, for example
 * for List<int>, you would use ":Add(int)".
 */
method = mono_class_get_method_from_name (clazz, "Add", 1);
mono_runtime_invoke (method, obj, args, &exception);

残念ながら、これは完全な例を示していないため、あまり役に立ちません。また、何をしても (使用するかどうかに関係mono_object_get_classなく)、ジェネリック メソッドを呼び出すと、警告が表示されてクラッシュします。

* Assertion: should not be reached at marshal.c:4315
SIGABRT stracktrace...

サンプルの C# クラスとそれを呼び出す C コードの下。非ジェネリック メソッドの呼び出しは問題なく機能しますが、ジェネリック メソッドの呼び出し方法がわかりません。役立つヒントがあれば幸いです。

サンプルCSharp

namespace foo {
  class MainClass {
      public static void Main(string[] args) {
              Console.WriteLine("Main");
      }

      public void GenericMember<T>(T t) {
             Console.WriteLine(t);
      }

      public static void GenericStatic<T>(T t) {
             Console.WriteLine(t);
      }


      public void NonGenericMember(string t) {
             Console.WriteLine(t);
      }

      public static void NonGenericStatic(string t) {
            Console.WriteLine(t);
      } 
  }
}

サンプル C

#include <mono/jit/jit.h>
#include <mono/metadata/mono-config.h>
#include <mono/metadata/metadata.h>
#include <mono/metadata/appdomain.h>
#include <mono/metadata/class.h> 
#include <mono/metadata/assembly.h>
#include <mono/metadata/image.h>
#include <mono/metadata/object.h>
#include <mono/metadata/debug-helpers.h>

int main(int argc, char **argv) {
   mono_config_parse(NULL);

   MonoDomain *domain = mono_jit_init_version("app", "v4.0.30319");
   MonoAssembly *assembly = mono_domain_assembly_open(domain, "foo.exe");  
   MonoImage *image = mono_assembly_get_image(assembly);  

   mono_jit_exec(domain, assembly, argc, argv);

   MonoClass *klass = mono_class_from_name(image, "foo", "MainClass");
   MonoObject *instance = mono_object_new(domain, klass)

   MonoString *string = mono_string_new(domain, "hello");
   void *params[] = { string };

   //NonGenericStatic call (works)
   MonoMethod *method = mono_class_get_method_from_name(klass, "NonGenericStatic", 1);
   mono_runtime_invoke(method, NULL, params, NULL);

   //NonGenericMember call (works);
   method = mono_class_get_method_from(klass, "NonGenericMember", 1);
   mono_runtime_invoke(method, instance, params, NULL);

   //GenericStatic call (fails)
   method = mono_class_get_method_from_name(klass, "GenericStatic", 1);
   mono_runtime_invoke(method, NULL, params, NULL);

   //GenericMember call (fails)
   method = mono_class_get_method_from_name(klass, "GenericMember", 1);
   mono_runtime_invoke(method, instance, params, NULL);

   mono_jit_cleanup(domain, assembly);

   return 0;
}
4

1 に答える 1

3

そのように汎用メソッドを呼び出すことはできません。残念ながら、ドキュメントは少し誤解を招くものです。public void GenericMember<T>(T t)パラメータを使用して呼び出したい場合はString、署名付きのメソッドを見つけてpublic void GenericMember<String>(String t)使用する必要がありmono_runtime_invokeます。

これを行う簡単な方法は、C# ユーティリティ メソッドを介してMethodInfo.MakeGenericMethodメソッドを使用することです。

public static IntPtr Utility(MethodInfo method, Type type)
{
  return method.MakeGenericMethod(type).MethodHandle.Value;
}

組み込み Mono API を使用してメソッドを見つけ、メソッドを表すメソッドと必要なタイプ (例では String)を使用してUtility呼び出します。この呼び出しの戻り値は、を表す になります。これは、呼び出す必要があるメソッドです。MonoMethod*GenericMethod<T>(T t)MonoMethod*GenericMethod<String>(String t)

ここにソース コードを含む詳細な投稿を作成しました。

于 2015-04-23T14:31:07.280 に答える