From: 011netservice@gmail.com Date: 2022-04-24 Subject: x86x64AnyCPU差異.txt 程式編譯成 x86, x64, Any CPU 的差異 ? ---------- 20200325 摘要 在 32 bit OS 上執行 ------------------------------------ ------------------------------------------------------------------- 1. Any CPU : 會用 32 bit Process 來執行,可以載入 Any CPU 和 x86 方式編譯的組件,但如果載入 x64 編譯的組件將會引發 BadImageFormatException 2. (Any CPU + 勾選建議使用32位元) : 同上. 3. x86 : 同上 4. x64 : 執行錯誤 = BadImageFormatException. 在 64 bit OS 上執行 ------------------------------------ ------------------------------------------------------------------- 1. Any CPU : 會用 64 bit Process 來執行,可以載入 Any CPU 和 x64 方式編譯的組件,但如果載入 x86 編譯的組件將會引發 BadImageFormatException 2. (Any CPU + 勾選建議使用32位元) : 會用 32 bit Process 來執行,可以載入 Any CPU 和 x86 方式編譯的組件,但如果載入 x64 編譯的組件將會引發 BadImageFormatException 3. x86 : 同 (Any CPU + 勾選建議使用32位元) 4. x64 : 同 Any CPU 勾選 Prefer 32-bit (建議使用32位元) AnyCPU + Prefer 32-bit 在 Windows 平台永遠跑 32 位元模式,跟平台目標設成 x86 的行為相同,唯一差別在於 AnyCPU + Perfer 32-bit 可以在 ARM 機器執行。參考: What AnyCPU Really Means As Of .NET 4.5 and Visual Studio 11 換句話說,AnyCPU + Prefer 32-bit 骨子裡根本是 x86,取消 Prefer 32-bit 後才是我原本想像的那個 32/64 都能跑的 AnyCPU。 下回看到 .NET 4.5+ 設定 AnyCPU 時,記得要確認沒有勾選 Prefer 32-bit,才代表 .NET 程式能 32/64 通吃。 Machine Config 依據執行的 Process 位元決定 32 bit Process 吃的 Machine Config 位置 : C:\Windows\Microsoft.NET\ Framework \v4.0.30319\Config 64 bit Process 吃的 Machine Config 位置 : C:\Windows\Microsoft.NET\ Framework64 \v4.0.30319\Config 怎麼查是用哪種方式編譯的? 可以透過 CorFlags.exe 的工具來查詢,該程式放在 C:\Program Files (x86)\Microsoft SDKs\Windows\v8.1A\bin\NETFX 4.5.1 Tools 路徑底下,設定為環境變數後就可以用 command 來檢視 PE $ corflags yourPE.exe PE : Process Executables (EXEs and DLLs) 查詢結果說明如下 Version : Assembly's target framework Header : 2.0/2.5 (Must have version of 2.5 or greater to run natively) PE : PE32 (32-bit)/PE32+ (64-bit) CorFlags : Hexadecimal value, computed base on below 4 flags. ILONLY : 1 if MSIS otherwise 0. 32BITREQ : 1 if 32-bit x86 only assembly otherwise 0. 32BITREF : 1 if 32-bit x86 only prefered in Any CPU architecture otherwise 0. 勾選 Prefer 32-bit (建議使用32位元)後=1 Signed : 1 if signed with strong name otherwise 0. 程式如何判斷是在 32 Bit 或 64 bit 執行中 ? 1. Environment.Is64BitProcess: Determines whether the current process is a 64-bit process. 2. Environment.Is64BitOperatingSystem: Determines whether the current operating system is a 64-bit operating system. 3. IntPtr.Size, 若為8, 則為 64bit 執行環境, 若為4, 則為 32bit 執行環境. ---------- 20200324 https://blog.darkthread.net/blog/anycpu-32bitpref/ 【茶包射手日記】只能跑 32 位元的 AnyCPU .NET 程式 測試某個 COM+ 元件應用專案,開發者所附的範例專案測試成功,我自己新增 Console Application 或 Windows Form 專案則卡在找不到 Registry 無法執行。強烈懷疑與 x86/x64 有關,由於只有註冊 64 位元 COM+,專案跑 x86 找不到 Registry 是意料中事,但詭異之處在於我已確認過範例專案跟我新增的專案都是設 Any CPU 無誤,甚至放在同一個 Solution 測試,卻一個成功一個失敗。 實測將新增 WinForm 或 Console 專案平台目標(Platform Target)改為 x64 可排除問題,但無法解答為什麼範例專案設 Any CPU 可以,我卻得寫死 x64 才行的疑惑。 挖出 CorFlags 工具一探究竟,發現一個奇妙差異,有個沒學過的新旗標: 32BITPREF,我新增的專案被設為 1,範例專案則為 0。 有了這項新發現,重新檢查 Visual Studio 的專案設定,這才看出玄機,兩個專案都設定 Any CPU, 但範例專案沒勾選 Prefer 32-bit,我新增的專案有。 用關鍵字查詢,我學到了新知識。參考: Make sure "Prefer 32-bit" option is turned off for .NET 4.5 executables 原來這是 Visual Studio 2012 起針對 .NET 4.5 專案的新增選項,在新增 WinForm、WPF、Console 專案時,預設為 Any CPU + Prefer 32-bit。 AnyCPU + Prefer 32-bit 在 Windows 平台永遠跑 32 位元模式,跟平台目標設成 x86 的行為相同,唯一差別在於 AnyCPU + Perfer 32-bit 可以在 ARM 機器執行。參考: What AnyCPU Really Means As Of .NET 4.5 and Visual Studio 11 換句話說,AnyCPU + Prefer 32-bit 骨子裡根本是 x86,取消 Prefer 32-bit 後才是我原本想像的那個 32/64 都能跑的 AnyCPU。 下回看到 .NET 4.5+ 設定 AnyCPU 時,記得要確認沒有勾選 Prefer 32-bit,才代表 .NET 程式能 32/64 通吃。 ---------- 20200324 https://toyo0103.github.io/2019/02/12/anycpu_x86_x64/ 將程式編譯成 x86、x64、Any CPU差別在哪 2019-02-12 | 0 Comments 相信用過 Visual Studio 的應該都注意過編譯時可以選擇 Any CPU、x86、x64 (x86、x64 需要編輯組態檔才會看的到) 這次踩到問題是,明明程式用 Any CPU 來編譯,但在 64 bit OS 上卻會用 32 bit Process 來執行 原來是因為專案建置的屬性勾到了 建議使用32位元的選項導致 解決的同時也引起了對於編譯成 x86、x64、Any CPU 有什麼差別的好奇心,找了文章後整理如下 差異 在 32 bit OS 上執行 Any CPU : 會用 32 bit Process 來執行,可以載入 Any CPU 和 x86 方式編譯的組件,但如果載入 x64 編譯的組件將會引發 BadImageFormatException Any CPU (勾選建議使用32位元) : 運作方式同上 x86 : 運作方式同上 x64 : 引發 BadImageFormatException 在 64 bit OS 上執行 Any CPU : 會用 64 bit Process 來執行,可以載入 Any CPU 和 x64 方式編譯的組件,但如果載入 x86 編譯的組件將會引發 BadImageFormatException Any CPU (勾選建議使用32位元) : 會用 32 bit Process 來執行,可以載入 Any CPU 和 x86 方式編譯的組件,但如果載入 x64 編譯的組件將會引發 BadImageFormatException x86 : 運作方式同 Any CPU (勾選建議使用32位元) x64 : 運作方式同 Any CPU Machine Config 依據執行的 Process 位元決定 32 bit Process 吃的 Machine Config 位置 : C:\Windows\Microsoft.NET\ Framework \v4.0.30319\Config 64 bit Process 吃的 Machine Config 位置 : C:\Windows\Microsoft.NET\ Framework64 \v4.0.30319\Config 怎麼查是用哪種方式編譯的? 可以透過 CorFlags.exe 的工具來查詢,該程式放在 C:\Program Files (x86)\Microsoft SDKs\Windows\v8.1A\bin\NETFX 4.5.1 Tools 路徑底下,設定為環境變數後就可以用 command 來檢視 PE $ corflags yourPE.exe PE : Process Executables (EXEs and DLLs) 查詢結果說明如下 Version : Assembly's target framework Header : 2.0/2.5 (Must have version of 2.5 or greater to run natively) PE : PE32 (32-bit)/PE32+ (64-bit) CorFlags : Hexadecimal value, computed base on below 4 flags. ILONLY : 1 if MSIS otherwise 0. 32BITREQ : 1 if 32-bit x86 only assembly otherwise 0. 32BITREF : 1 if 32-bit x86 only prefered in Any CPU architecture otherwise 0. Signed : 1 if signed with strong name otherwise 0. 意義如下 (圖片節錄至 : https://stackoverflow.com/questions/18608785/how-to-interpret-the-corflags-flags/23614024#23614024)