为什么 EnumDomains/NextDomain 永远循环?

Why do EnumDomains/NextDomain loop forever?

下面的简单代码可以直接运行在Visual StudioC++控制台工程中

它将永远循环,因为 NextDomain 将始终 return 相同的 IUnknown *

根据 Microsoft 的说法,如果枚举结束,它应该 return NULL。参见 https://docs.microsoft.com/en-us/dotnet/framework/unmanaged-api/hosting/icorruntimehost-nextdomain-method

但是,从未达到 NULL。它似乎从不 return NextDomain,而是保持 return 同一个域。

有什么问题吗?谢谢

#include <iostream>
#include <Windows.h>
#include <metahost.h>
#include <mscoree.h>
#pragma comment(lib, "mscoree.lib")

int main()
{
    ICLRMetaHost* clrMetaHost = NULL;
    ICLRRuntimeInfo* clrRuntimeInfo = NULL;
    ICorRuntimeHost* clrCorRuntimeHost = NULL;

    do {
        if (FAILED(CLRCreateInstance(CLSID_CLRMetaHost, IID_PPV_ARGS(&clrMetaHost))))
        {
            std::cout << "failed 1" << std::endl;
            break;
        }


        if (FAILED(clrMetaHost->GetRuntime(L"v2.0.50727", IID_PPV_ARGS(&clrRuntimeInfo))))
        {
            if (FAILED(clrMetaHost->GetRuntime(L"v4.0.30319", IID_PPV_ARGS(&clrRuntimeInfo))))
            {
                std::cout << "failed 2" << std::endl;
                break;
            }
        }

        if (FAILED(clrRuntimeInfo->GetInterface(CLSID_CorRuntimeHost, IID_PPV_ARGS(&clrCorRuntimeHost))))
        {
            std::cout << "failed 3" << std::endl;
            break;
        }


        if (FAILED(clrCorRuntimeHost->Start()))
        {
            std::cout << "failed 4" << std::endl;
            break;
        }

        HDOMAINENUM hDomainEnum = nullptr;
        if (FAILED(clrCorRuntimeHost->EnumDomains(&hDomainEnum))) {
            std::cout << "failed 5" << std::endl;
            break;
        }

        IUnknown* domain;
        while (SUCCEEDED(clrCorRuntimeHost->NextDomain(hDomainEnum, &domain))
            && domain != NULL) {
            std::cout << "why loop forever here?" << std::endl;
            domain->Release();
        }

        if (FAILED(clrCorRuntimeHost->CloseEnum(hDomainEnum))) {
            std::cout << "failed 6" << std::endl;
        }

        break;

    } while (0);

    if (clrCorRuntimeHost != NULL) clrCorRuntimeHost->Release();
    if (clrRuntimeInfo != NULL) clrRuntimeInfo->Release();
    if (clrMetaHost != NULL) clrMetaHost->Release();

    return 0;
}
大多数枚举器的

Return 代码是 S_OK 继续和 S_FALSE 当它没有成功时。 S_FALSE 不是失败代码。

while (S_OK == clrCorRuntimeHost->NextDomain(hDomainEnum, &domain))
            && domain != NULL) {
            std::cout << "why loop forever here?" << std::endl;
            domain->Release();
        }