单声道嵌入,从 C 调用 C# 泛型方法



我在弄清楚如何从 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;
}

您不能以这种方式调用泛型方法,不幸的是,文档有点误导。如果要使用String参数调用public void GenericMember<T>(T t),则必须找到带有签名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 查找Utility方法,使用表示GenericMethod<T>(T t)方法和所需类型的MonoMethod*调用它(示例中为字符串)。此调用的返回将是一个表示GenericMethod<String>(String t)MonoMethod*。这是您需要调用的方法。

我在这里用源代码做了一个详细的帖子。

最新更新