為C# as 類型轉換及Assembly.LoadFrom埋坑!

来源:http://www.cnblogs.com/cyq1162/archive/2016/12/22/6212200.html
-Advertisement-
Play Games

不久前,我發佈了一個調試工具:發佈:.NET開發人員必備的可視化調試工具(你值的擁有)不久前,我發佈了一個調試工具:發佈:.NET開發人員必備的可視化調試工具(你值的擁有) 效果是這樣的:之後,有小部分用戶反映,工具不生效~~~然後,建議小部分用戶換個電腦環境試試,就好了~~~於是... ...


背景:

不久前,我發佈了一個調試工具:發佈:.NET開發人員必備的可視化調試工具(你值的擁有)

效果是這樣的:

之後,有小部分用戶反映,工具用不了(沒反應或有異常)~~~

然後,建議小部分用戶換個電腦環境試試,有些就好了~~~

於是,我假定是VS環境下的 Microsoft.VisualStudio.DebuggerVisualizers.dll 的版本不一致引發的。

因此,一般我都建議用戶自己下載源碼,重新引用去編繹一下!!!

由於該工具一直在CSDN論壇的VB.NET版塊置頂著。

考慮到受眾多,中間還偷偷升級過幾回,解決了拋異常的問題,不過仍沒有從根本性解決~~~~

這兩天,有個叫子寒的同學,找上了我,希望我幫他解決這個問題。

我試著重新編繹了新版本發給他,都反饋木有效果。

只好讓他下載源碼,併在他電腦上進行遠程調試。

昨晚處理到深夜1點半,終於:把發現的兩個坑給埋了!!!

下麵介紹下這兩個坑:

1:as 轉換的坑:

先看一段源碼,這是拿到反序列化的結果,轉Table,再綁定:

 protected override void Show(IDialogVisualizerService windowService, IVisualizerObjectProvider objectProvider)
        {
            MDataTable dt = objectProvider.GetObject() as MDataTable;
            FormCreate.BindTable(windowService, dt, null);
        }

在這段代碼中,調試的結果:

1:objectProvider.GetObject() 拿到的對象是MDataTable,GetType也返回的CYQ.Data.Table.MDataTable。

2as MDataTable 卻返回了null ?

咦?一個大大的問號在我面前,同樣的類型,怎麼as不過去?

於是我把代碼改了一下:

MDataTable dt=(MDataTable)objectProvider.GetObject()

拋異常了:

************** 異常文本 **************
System.InvalidCastException: [A]CYQ.Data.Table.MDataTable 無法強制轉換為 [B]CYQ.Data.Table.MDataTable。
類型 A 源自“CYQ.Data, Version=5.7.5.5, Culture=neutral, PublicKeyToken=null”(在位元組數組的上下文“LoadNeither”中)。
類型 B 源自“CYQ.Data, Version=5.7.5.5, Culture=neutral, PublicKeyToken=null”(在上下文“LoadFrom”中的“C:\Program Files (x86)\Microsoft Visual Studio 12.0\Common7\Packages\Debugger\Visualizers\CYQ.Data.dll”位置處)。 在 CYQ.Visualizer.MDataTableVisualizer.Show(IDialogVisualizerService windowService, IVisualizerObjectProvider objectProvider) 在 Microsoft.VisualStudio.DebuggerVisualizers.DebugViewerShim.ManagedShim.DelegatedHost.CreateViewer(IntPtr hwnd, HostServicesHelper hsh, SafeProxyWrapper proxy)

這個異常是什麼等會再說,先補充知識點先:

1as 類型轉換:只檢測上下文中類型是否一致(或存在隱式轉換),若失敗返回null,不拋異常。

2:強制類型轉換:嘗試進行類型轉換,轉換失敗時,拋出異常。

好吧,第一個坑,相同的類型,沒有異常,埋的夠深!!!

AS叫了:這坑不能怪我,要怪就怪Assembly.LoadFrom,誰讓你們把我們分隔在不同的上下文中。

2:Assembly.LoadFrom 的坑

這裡再貼一段詳細的異常信息:

mscorlib
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.NET/Framework/v4.0.30319/mscorlib.dll
----------------------------------------
Microsoft.VisualStudio.Platform.AppDomainManager
    程式集版本:12.0.0.0
    Win32 版本:12.0.21005.1
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/Microsoft.VisualStudio.Platform.AppDomainManager/v4.0_12.0.0.0__b03f5f7f11d50a3a/Microsoft.VisualStudio.Platform.AppDomainManager.dll
----------------------------------------
System
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System/v4.0_4.0.0.0__b77a5c561934e089/System.dll
----------------------------------------
System.Xml
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Xml/v4.0_4.0.0.0__b77a5c561934e089/System.Xml.dll
----------------------------------------
System.Configuration
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Configuration/v4.0_4.0.0.0__b03f5f7f11d50a3a/System.Configuration.dll
----------------------------------------
System.Windows.Forms
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Windows.Forms/v4.0_4.0.0.0__b77a5c561934e089/System.Windows.Forms.dll
----------------------------------------
System.Drawing
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Drawing/v4.0_4.0.0.0__b03f5f7f11d50a3a/System.Drawing.dll
----------------------------------------
Microsoft.VisualStudio.DebuggerVisualizers
    程式集版本:12.0.0.0
    Win32 版本:12.0.21005.1
    基本代碼:file:///C:/Windows/assembly/GAC_MSIL/Microsoft.VisualStudio.DebuggerVisualizers/12.0.0.0__b03f5f7f11d50a3a/Microsoft.VisualStudio.DebuggerVisualizers.dll
----------------------------------------
CYQ.Visualizer
    程式集版本:2.0.0.5
    Win32 版本:2.0.0.5
    基本代碼:file:///C:/Program%20Files%20(x86)/Microsoft%20Visual%20Studio%2012.0/Common7/Packages/Debugger/Visualizers/CYQ.Visualizer.dll
----------------------------------------
CYQ.Data
    程式集版本:5.7.5.5
    Win32 版本:5.7.5.5
    基本代碼:file:///C:/Program%20Files%20(x86)/Microsoft%20Visual%20Studio%2012.0/Common7/Packages/Debugger/Visualizers/CYQ.Data.DLL
----------------------------------------
System.Data
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_32/System.Data/v4.0_4.0.0.0__b77a5c561934e089/System.Data.dll
----------------------------------------
System.Core
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Core/v4.0_4.0.0.0__b77a5c561934e089/System.Core.dll
----------------------------------------
CYQ.Data
    程式集版本:5.7.5.5
    Win32 版本:12.0.21005.1
    基本代碼:file:///C:/Windows/assembly/GAC_MSIL/Microsoft.VisualStudio.DebuggerVisualizers/12.0.0.0__b03f5f7f11d50a3a/Microsoft.VisualStudio.DebuggerVisualizers.dll
----------------------------------------
System.Numerics
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Numerics/v4.0_4.0.0.0__b77a5c561934e089/System.Numerics.dll
----------------------------------------
System.Transactions
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_32/System.Transactions/v4.0_4.0.0.0__b77a5c561934e089/System.Transactions.dll
----------------------------------------
System.EnterpriseServices
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.36373 built by: FX452RTMLDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_32/System.EnterpriseServices/v4.0_4.0.0.0__b03f5f7f11d50a3a/System.EnterpriseServices.dll
----------------------------------------
System.Windows.Forms.resources
    程式集版本:4.0.0.0
    Win32 版本:4.0.30319.34209 built by: FX452RTMGDR
    基本代碼:file:///C:/Windows/Microsoft.Net/assembly/GAC_MSIL/System.Windows.Forms.resources/v4.0_4.0.0.0_zh-Hans_b77a5c561934e089/System.Windows.Forms.resources.dll
----------------------------------------
************** 已載入的程式集 **************

 異常里核心的兩段:

CYQ.Visualizer
    程式集版本:2.0.0.5
    Win32 版本:2.0.0.5
    基本代碼:file:///C:/Program%20Files%20(x86)/Microsoft%20Visual%20Studio%2012.0/Common7/Packages/Debugger/Visualizers/CYQ.Visualizer.dll
----------------------------------------
CYQ.Data
    程式集版本:5.7.5.5
    Win32 版本:5.7.5.5
    基本代碼:file:///C:/Program%20Files%20(x86)/Microsoft%20Visual%20Studio%2012.0/Common7/Packages/Debugger/Visualizers/CYQ.Data.DLL

CYQ.Data
    程式集版本:5.7.5.5
    Win32 版本:12.0.21005.1
    基本代碼:file:///C:/Windows/assembly/GAC_MSIL/Microsoft.VisualStudio.DebuggerVisualizers/12.0.0.0__b03f5f7f11d50a3a/Microsoft.VisualStudio.DebuggerVisualizers.dll

從上面的異常錯誤信息中,第一眼看是要蒙B的,畢竟少見多怪啊~~~~

大概花了8分鐘,終於反應過來了,經過仔細的思考,可以發現:

1:CYQ.Data 被CYQ.Visualizer.dll載入是在同一目錄下。

2:被Microsoft.VisualStudio.DebuggerVisualizers.dll載入卻是在(上下文“LoadNeither”中,鬼知道這個LoadNeither是什麼)。

於是,同一個dll被載入成兩個不同路徑,導致上下文環境不一樣,而不能互轉~~~~

為什麼大部分正常,小部分環境會這麼處理呢,目前無從知道,微軟也是愛造坑的~~~

好吧,坑已經知道了,接下來如何埋才是重點:

1:埋坑思維一:測試下Assembly.Load、Assembly.LoadFile、Assembly.LoadFrom

寫了幾段測試代碼:

 string filePath = AppDomain.CurrentDomain.BaseDirectory + "cyq\\CYQ.Data.dll";
            byte[] bytes = File.ReadAllBytes(filePath);
            Assembly ass = Assembly.Load(bytes);
            object o = ass.CreateInstance("CYQ.Data.Table.MDataTable");
 MDataTable dt = (MDataTable)o;

失敗!

 string filePath = AppDomain.CurrentDomain.BaseDirectory + "cyq\\CYQ.Data.dll";
 Assembly ass = Assembly.LoadFrom(filePath);
 Assembly.LoadFile(filePath);
...省略...

都失敗!!

測試這三個方法,主要是想看看有沒有啥本質的不一樣,特別是和強簽名合在一起後~~~

把dll加個強簽名試試~~~

仍失敗~~~~

看來,我想多了,不同的dll路徑載入的對象轉換這條路是不通的了~~~~

2:埋坑思維二:強簽名DLL,並註冊到GAC中

理論上來說:該方式100%是可行的,畢竟路徑的引用都是一致的。

方式也簡單:通過代碼加個註冊的命令也不是什麼難事~~~~

還是思考有沒有其它更簡潔的方式!

3:埋坑思維三:找個穩定的中介

既然問題出現在序列化前的MDataTable和反序列化後的MDataTable在不同上下文的dll而導致的。

那就就找一種第三方中介了:MDataTable=>中介(序列化)=》中介(反序列化)=>MDataTable

不用動腦,就可以想到兩種:json或 DataTable。

於是代碼就動起來了:

序列化時,用DataTable

反序列化時:

打完收工,重新編繹,發給用戶測試,正常通過~

工具下載地址:

http://www.cnblogs.com/cyq1162/p/6027051.html

總結:

1:這麼多年,第一次栽坑在as轉換上,也許是沒想到,也許是萬萬想不到。

2:不同的路徑載入的相同的dll類型無法互轉,這麼多年終於遇上了,說明上的山多還是會見鬼的。

可既然版本號和簽名都一致,又認為簽名無法偽造,那為什麼不呢?

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 中國科學技術大學:   http://mirrors.ustc.edu.cn/ (IPv4+IPv6)   http://mirrors4.ustc.edu.cn/   http://mirrors6.ustc.edu.cn/   西南大學:http://linux.swu.edu.cn/s... ...
  • 說來慚愧,玩Linux這麼久了,居然沒有玩轉vmstat這個命令,對很多指標的具體意義都有點模糊不清,花了點時間好好學習、整理一下這個命令的相關資料。因為這個命令確實比較重要,而且頻繁用到。 命令簡介: vmstat是Virtual Meomory Statistics(虛擬記憶體統計)的縮寫,可對操... ...
  • 一、概要 由於監控業務發展的較快,各種告警很多,並且告警記錄不能查詢,則需要一個平臺來解決告警展示、查詢等問題,「統一告警平臺」應運而生。以下簡稱AMC(Alert Messages Center)。 AMC提供介面調用和前臺配置,支持Rtx、簡訊、微信、郵件四種告警通道,支持模塊信息自動補全、告警 ...
  • 如果你是一個Linux新手,並且剛剛安裝了一個新的英文系統但想要設置成中文系統,肯定會接觸到上面幾個變數,在網上搜索了一系列解決方法,給一些變數賦一下值,再export一下,或者寫到配置文件裡面,然後就搞定了,但究竟為什麼要這樣做,可能還是一知半解。通過這兩天自己對網上看到的一些文章的整理和自己的試 ...
  • private void Form1_FormClosing(object sender, FormClosingEventArgs e) { DialogResult re = MessageBox.Show("確定離開嗎","詢問",MessageBoxButtons.YesNo,Message ...
  • private void button1_Click(object sender, EventArgs e) { MessageBox.Show("我也喜歡你!"); //if (this.button1.Text == "喜歡") //{ // button2.Visible = false; / ...
  • 靜態用戶名和密碼的登錄練習 private void button2_Click(object sender, EventArgs e) { textUser.Text = Convert.ToString(textUser.Text); textBox2.Text = textBox2.Text. ...
  • 數組、集合、異常捕獲 數組: 一次性存儲多個相同類型的變數。 一維數組: 語法: 數組類型[] 數組名=new 數組類型[數組長度]; 聲明數組的語法: A.數據類型 [] 數組名稱= new 數據類型[2]{1,2}; B.數據類型 [] 數組名稱 = new 數據類型[數組大小]; C. 數據類 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...