如何在托管CLR的本地c++应用程序中创建泛型.net类型(不使用c++ /CLI)

How to create a generic .NET type in a native c++ application hosting the CLR (not using C++/CLI)?

本文关键字:c++ 类型 net 泛型 CLI 创建 CLR 应用程序      更新时间:2023-10-16

在。net程序中,我可以通过以下方式创建泛型类型:

 System::Type::MakeGenericType(...)

必须有一种方法在本机c++中为。net类型(_TypePtr)做到这一点。我托管我自己的CLR实例,不使用c++/CLI。(这里有一个如何完成的示例。)我的方法是:

_TypePtr BuildGenericType(_TypePtr spGenericType, _TypePtr spTypeArgs[]) 
{
    return spGenericType-> ..... ???
}

但是没有像MakeGenericType这样的方法,我不知道在哪里可以找到它。对如何解决这个问题有什么想法吗?

最后我自己找到了一个解决办法。首先,我需要一个额外的程序集来包装System::TypeBuilder

/// <summary>
/// Wrapper for System::TypeBuilder
/// </summary>
public class TypeBuilder
{
    /// <summary>
    /// Creates a generic type out of the given arguments.
    /// </summary>
    /// <param name="generic">The unqualified generic type.</param>
    /// <param name="typeArgs">The typearguments for the generic type.</param>
    /// <returns>The qualified generic type.</returns>
    public static Type MakeGenericType(Type generic, params Type[] typeArgs)
    {
        return generic.MakeGenericType(typeArgs);
    }
}

从c++

加载的附加程序集

我在那里运行了这些性感的行,其中CLRAssembly是clr的抽象层。

_TypePtr CLRAssembly::BuildType(_TypePtr spGenericType, _TypePtr spTypeArgs[]) 
{
    LONG index = 0;
    SAFEARRAY* psaArgs = SafeArrayCreateVector(VT_VARIANT, 0, 1 + (sizeof(spTypeArgs)/sizeof(_TypePtr)));
    SafeArrayPutElement(psaArgs, &index, &_variant_t((IUnknown*)spGenericType, true));
    for(int i = 0; i < sizeof(spTypeArgs)/sizeof(_TypePtr); i++) {
        index++;
        SafeArrayPutElement(psaArgs, &index, &_variant_t((IUnknown*)spTypeArgs[i], true));
    }
    return (_TypePtr)clraAssemblyHelper->RunMethod(L"AssemblyHelper.TypeBuilder", L"MakeGenericType", psaArgs); 
}

最后我可以像这样运行我的示例方法:

DDDElements::Defs* ListDefs::AddNew()
{
    _TypePtr params[1];
    params[0] = clra->BuildType(L"DDD.Elements.Defs");
    _TypePtr spType = clra->BuildType(clra->BuildType(L"DDD.ElementList`1"), params);
    return new DDDElements::Defs(clr, clra, 
        clra->RunMethod(spType, vtCLRObject, L"AddNew")
    );
}

最后它工作得很好。:)

/解决