轉載自:http://www.linuxidc.com/Linux/2012-10/72293.htm
機制介紹
有些時候主程序是經過C/C++實現的,可是咱們但願經過託管代碼來擴展非託管程序,從而也得到託管代碼帶來的一系列優勢。好比開發效率高,自動垃圾回收等。linux
運行託管與非託管代碼根本區別在於託管代碼是進程首先加載CLR而後經過CLR運行託管程序,而非託管代碼則是操做系統直接根據其PE Header加載程序分配內存從而運行。所以若是須要經過託管代碼來擴展非託管程序,首先要加載CLR來使非託管程序得到運行託管代碼的能力。windows
可使用如下過程將 CLR 加載到進程中:函數
- 調用 CLRCreateInstance 函數以獲取 ICLRMetaHost 或 ICLRMetaHostPolicy 接口。 CLRCreateInstance 函數取代 .NET Framework 1.1 和 2.0 承載全局靜態函數部分中列出的全部 CorBindTo* 函數。
- 調用 ICLRMetaHost::EnumerateInstalledRuntimes、ICLRMetaHost::GetRuntime 或 ICLRMetaHostPolicy::GetRequestedRuntime 方法以獲取有效的 ICLRRuntimeInfo 指針。
- 調用 ICLRRuntimeInfo::GetInterface 方法。 爲 rclsid 參數指定 CLSID_CLRRuntimeHost,併爲 riid 參數指定 IID_ICLRRuntimeHost。
全部這些接口的原型均位於 Metahost.h 文件中,該文件位於 Windows 軟件開發工具包 (SDK) 的 Include 目錄中。 宿主可使用 ICLRRuntimeInfo 和 ICLRRuntimeHost 接口來控制要加載哪一個版本的運行時以及基本功能(如垃圾回收和程序集加載)的行爲。使用 ICLRRuntimeHost 接口能夠執行如下操做:工具
- 經過調用 ICLRRuntimeHost::Start 方法來啓動運行時。
- 執行託管代碼。
- 獲取指向 ICLRControl 接口(可提供對由公共語言運行時實現的管理器的訪問)的指針,以及註冊實現 IHostControl 接口的宿主控件對象。 公共語言運行時調用 IHostControl 接口來肯定宿主實現的管理器。
參考這裏 http://msdn.microsoft.com/en-us/library/01918c6x.aspx開發工具
實例代碼spa
如下是C++加載CLR運行託管程序的實例代碼,啓動CLR以後經過調用ExecuteInDefaultAppDomain來運行託管程序SampleManagedApp.exe中名爲Test的程序。這裏要注意的是ExecuteInDefaultAppDomain只能執行託管代碼簽名爲static int pwzMethodName (String pwzArgument)的方法。操作系統
- #include <SDKDDKVer.h>
-
- #include <stdio.h>
- #include <tchar.h>
- #include <windows.h>
-
- #include <metahost.h>
- #include <mscoree.h>
- #pragma comment(lib, "mscoree.lib")
-
- int _tmain(int argc, _TCHAR* argv[])
- {
- ICLRMetaHost *pMetaHost = nullptr;
- ICLRMetaHostPolicy *pMetaHostPolicy = nullptr;
- ICLRRuntimeHost *pRuntimeHost = nullptr;
- ICLRRuntimeInfo *pRuntimeInfo = nullptr;
-
- HRESULT hr = CLRCreateInstance(CLSID_CLRMetaHost, IID_ICLRMetaHost, (LPVOID*)&pMetaHost);
- hr = pMetaHost->GetRuntime(L"v4.0.30319", IID_PPV_ARGS(&pRuntimeInfo));
-
- if(FAILED(hr))
- {
- goto cleanup;
- }
-
- hr = pRuntimeInfo->GetInterface(CLSID_CLRRuntimeHost, IID_PPV_ARGS(&pRuntimeHost));
-
- hr = pRuntimeHost->Start();
-
- DWORD dwRet = 0;
- hr = pRuntimeHost->ExecuteInDefaultAppDomain(L"SampleManagedApp.exe",
- L"SampleManagedApp.Program",
- L"Test",
- L"Hello World!",
- &dwRet);
-
- hr = pRuntimeHost->Stop();
-
- cleanup:
- if(pRuntimeInfo != nullptr)
- {
- pRuntimeInfo->Release();
- pRuntimeInfo = nullptr;
- }
-
- if(pRuntimeHost != nullptr)
- {
- pRuntimeHost->Release();
- pRuntimeHost = nullptr;
- }
-
- if(pMetaHost != nullptr)
- {
- pMetaHost->Release();
- pMetaHost = nullptr;
- }
- }
相應的託管代碼以下,指針
- using System;
-
- namespace SampleManagedApp
- {
- class Program
- {
- static void Main(string[] args)
- {
- }
-
- public static int Test(string s)
- {
- Console.WriteLine(s);
- return 0;
- }
- }
- }
最終將SampleManagedApp.exe與非託管程序放在同一個路徑下運行非託管程序,就會獲得最終Console中輸出:Hello World!htm