裝箱、拆箱

来源:http://www.cnblogs.com/bb-love-dd/archive/2016/10/29/6010120.html
-Advertisement-
Play Games

下麵是我在網上收集的裝箱、拆箱的資料,先收集起來以後慢慢看,越往後看越有點蒙。。。 1、 裝箱和拆箱是一個抽象的概念 2、 裝箱是將值類型轉換為引用類型 ;拆箱是將引用類型轉換為值類型 利用裝箱和拆箱功能,可通過允許值類型的任何值與Object 類型的值相互轉換,將值類型與引用類型鏈接起來 例如: ...


下麵是我在網上收集的裝箱、拆箱的資料,先收集起來以後慢慢看,越往後看越有點蒙。。。

1、
      裝箱和拆箱是一個抽象的概念 
2、
      裝箱是將值類型轉換為引用類型 ;拆箱是將引用類型轉換為值類型 
      利用裝箱和拆箱功能,可通過允許值類型的任何值與Object 類型的值相互轉換,將值類型與引用類型鏈接起來 
例如: 
int val = 100; 
object obj = val; 
Console.WriteLine ("對象的值 = {0}", obj); 
這是一個裝箱的過程,是將值類型轉換為引用類型的過程 

int val = 100; 
object obj = val; 
int num = (int) obj; 
Console.WriteLine ("num: {0}", num); 
這是一個拆箱的過程,是將值類型轉換為引用類型,再由引用類型轉換為值類型的過程 

註:被裝過箱的對象才能被拆箱
3、
      .NET中,數據類型劃分為值類型和引用(不等同於C++的指針)類型,與此對應,記憶體分配被分成了兩種方式,一為棧,二為堆,註意:是托管堆。
      值類型只會在棧中分配。
      引用類型分配記憶體與托管堆。
      托管堆對應於垃圾回收。
4:裝箱/拆箱是什麼? 
裝箱:用於在垃圾回收堆中存儲值類型。裝箱是值類型到 object 類型或到此值類型所實現的任何介面類型的隱式轉換。 
拆箱:從 object 類型到值類型或從介面類型到實現該介面的值類型的顯式轉換。 
5:為何需要裝箱?(為何要將值類型轉為引用類型?) 
一種最普通的場景是,調用一個含類型為Object的參數的方法,該Object可支持任意為型,以便通用。當你需要將一個值類型(如Int32)傳入時,需要裝箱。 
另一種用法是,一個非泛型的容器,同樣是為了保證通用,而將元素類型定義為Object。於是,要將值類型數據加入容器時,需要裝箱。 
6:裝箱/拆箱的內部操作。 
裝箱: 
對值類型在堆中分配一個對象實例,並將該值複製到新的對象中。按三步進行。 
第一步:新分配托管堆記憶體(大小為值類型實例大小加上一個方法表指針和一個SyncBlockIndex)。 
第二步:將值類型的實例欄位拷貝到新分配的記憶體中。 
第三步:返回托管堆中新分配對象的地址。這個地址就是一個指向對象的引用了。 
有人這樣理解:如果將Int32裝箱,返回的地址,指向的就是一個Int32。我認為也不是不能這樣理解,但這確實又有問題,一來它不全面,二來指向Int32並沒說出它的實質(在托管堆中)。 
拆箱:
檢查對象實例,確保它是給定值類型的一個裝箱值。將該值從實例複製到值類型變數中。 
有書上講,拆箱只是獲取引用對象中指向值類型部分的指針,而內容拷貝則是賦值語句之觸發。我覺得這並不要緊。最關鍵的是檢查對象實例的本質,拆箱和裝箱的類型必需匹配,這一點上,在IL層上,看不出原理何在,我的猜測,或許是調用了類似GetType之類的方法來取出類型進行匹配(因為需要嚴格匹配)。 
7:裝箱/拆箱對執行效率的影響 
顯然,從原理上可以看出,裝箱時,生成的是全新的引用對象,這會有時間損耗,也就是造成效率降低。 
那該如何做呢? 
首先,應該儘量避免裝箱。 
比如上例2的兩種情況,都可以避免,在第一種情況下,可以通過重載函數來避免。第二種情況,則可以通過泛型來避免。 
當然,凡事並不能絕對,假設你想改造的代碼為第三方程式集,你無法更改,那你只能是裝箱了。 
對於裝箱/拆箱代碼的優化,由於C#中對裝箱和拆箱都是隱式的,所以,根本的方法是對代碼進行分析,而分析最直接的方式是瞭解原理結何查看反編譯的IL代碼。比如:在迴圈體中可能存在多餘的裝箱,你可以簡單採用提前裝箱方式進行優化。 
8:對裝箱/拆箱更進一步的瞭解 
裝箱/拆箱並不如上面所講那麼簡單明瞭,比如:裝箱時,變為引用對象,會多出一個方法表指針,這會有何用處呢? 
我們可以通過示例來進一步探討。 
舉個例子。 
Struct A : ICloneable 
{ 
public Int32 x; 
public override String ToString() { 
return String.Format(”{0}”,x); 
} 
public object Clone() { 
return MemberwiseClone(); 
} 
} 
static void main() 
{ 
A a; 
a.x = 100; 
Console.WriteLine(a.ToString()); 
Console.WriteLine(a.GetType()); 
A a2 = (A)a.Clone(); 
ICloneable c = a2; 
Ojbect o = c.Clone(); 
} 
5.0:a.ToString()。編譯器發現A重寫了ToString方法,會直接調用ToString的指令。因為A是值類型,編譯器不會出現多態行為。因此,直接調用,不裝箱。(註:ToString是A的基類System.ValueType的方法) 
5.1:a.GetType(),GetType是繼承於System.ValueType的方法,要調用它,需要一個方法表指針,於是a將被裝箱,從而生成方法表指針,調用基類的System.ValueType。(補一句,所有的值類型都是繼承於System.ValueType的)。 
5.2:a.Clone(),因為A實現了Clone方法,所以無需裝箱。 
5.3:ICloneable轉型:當a2為轉為介面類型時,必須裝箱,因為介面是一種引用類型。 
5.4:c.Clone()。無需裝箱,在托管堆中對上一步已裝箱的對象進行調用。 
附:其實上面的基於一個根本的原理,因為未裝箱的值類型沒有方法表指針,所以,不能通過值類型來調用其上繼承的虛方法。另外,介面類型是一個引用類型。對此,我的理解,該方法表指針類似C++的虛函數表指針,它是用來實現引用對象的多態機制的重要依據。 
9:如何更改已裝箱的對象 
對於已裝箱的對象,因為無法直接調用其指定方法,所以必須先拆箱,再調用方法,但再次拆箱,會生成新的棧實例,而無法修改裝箱對象。有點暈吧,感覺在說繞口令。還是舉個例子來說:(在上例中追加change方法) 
public void Change(Int32 x) { 
this.x = x; 
} 
調用: 
A a = new A(); 
a.x = 100; 
Object o = a; //裝箱成o,下麵,想改變o的值。 
((A)o).Change(200); //改掉了嗎?沒改掉。 
沒改掉的原因是o在拆箱時,生成的是臨時的棧實例A,所以,改動是基於臨時A的,並未改到裝箱對象。 
(附:在托管C++中,允許直接取加拆箱時第一步得到的實例引用,而直接更改,但C#不行。) 
那該如何是好? 
嗯,通過介面方式,可以達到相同的效果。 
實現如下: 
interface IChange { 
void Change(Int32 x); 
} 
struct A : IChange { 
… 
} 
調用: 
((IChange)o).Change(200);//改掉了嗎?改掉了。 
為啥現在可以改? 
在將o轉型為IChange時,這裡不會進行再次裝箱,當然更不會拆箱,因為o已經是引用類型,再因為它是IChange類型,所以可以直接調用Change,於是,更改的也就是已裝箱對象中的欄位了,達到期望的效果。

10、--------------------------
      將值類型轉換為引用類型,需要進行裝箱操作(boxing):
1、首先從托管堆中為新生成的引用對象分配記憶體。
2、然後將值類型的數據拷貝到剛剛分配的記憶體中。
3、返回托管堆中新分配對象的地址。
可以看出,進行一次裝箱要進行分配記憶體和拷貝數據這兩項比較影響性能的操作。
將引用內型轉換為值內型,需要進行拆箱操作(unboxing):
1、首先獲取托管堆中屬於值類型那部分欄位的地址,這一步是嚴格意義上的拆箱。
2、將引用對象中的值拷貝到位於線程堆棧上的值類型實例中。
經過這2步,可以認為是同boxing是互反操作。嚴格意義上的拆箱,並不影響性能,但伴隨這之後的拷貝數據的操作就會同boxing操作中一樣影響性能。

11、-------------------------
NET的所有類型都是由基類System.Object繼承過來的,包括最常用的基礎類型:int, byte, short,bool等等,就是說所有的事物都是對象。如果申明這些類型得時候都在堆(HEAP)中分配記憶體,會造成極低的效率!(個中原因以及關於堆和棧得區別會在另一篇里單獨得說說!)
.NET如何解決這個問題得了?正是通過將類型分成值型(value)和引用型(regerencetype),C#中定義的值類型包括原類型(Sbyte、Byte、Short、Ushort、Int、Uint、Long、Ulong、Char、Float、Double、Bool、Decimal)、枚舉(enum)、結構(struct),引用類型包括:類、數組、介面、委托、字元串等。
值型就是在棧中分配記憶體,在申明的同時就初始化,以確保數據不為NULL;
引用型是在堆中分配記憶體,初始化為null,引用型是需要GARBAGE COLLECTION來回收記憶體的,值型不用,超出了作用範圍,系統就會自動釋放!
下麵就來說裝箱和拆箱的定義!
裝箱就是隱式的將一個值型轉換為引用型對象。比如:
int i=0;
Syste.Object obj=i;
這個過程就是裝箱!就是將i裝箱!
拆箱就是將一個引用型對象轉換成任意值型!比如:
int i=0;
System.Object obj=i;
int j=(int)obj;
這個過程前2句是將i裝箱,後一句是將obj拆箱!

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

-Advertisement-
Play Games
更多相關文章
  • <DOCTYPE html> <thml lang = "en"> <head> <meta charset = "UTF-8"> <title>空間</title> <style type = "text/css"> *{ padding : 0; margin : 0; } ul{ list-s ...
  • 大家好,我是Traceless,很高興能得到作者 幸運★星 大佬的授權,為大家整理框架使用教程及開發文檔,本文及立足於基礎科普,包含框架原理介紹、開發環境、使用環境搭建及日常框架使用說明。對開發老鳥來說,說許你只需要翻閱使用環境、插件使用和下載API介面文檔即可。 此處要對框架的版權等信息做一個說明 ...
  • 首先,實現環境VS2013+windows10。 程式界面預覽: 瀏覽器可以實現搜索,新建,關閉功能。簡單的基礎功能都能實現。 下麵是代碼: 從上到下的講解: 1.標題 2.頂部菜單 ToolStrip控制項 (1)步驟: 1)拖入ToolStirp控制項。 2)編輯ToolStrip選項button。 ...
  • using System; using System.Collections.Generic; using System.Linq; using System.Text; //泛型解決原來集合類中元素的裝箱和拆箱問題 namespace _012 { class Program { static v ...
  • 不同的瀏覽器會把cookie文件保存在不同的地方 以下是C# WebBrowser控制項cookies的存放路徑 C:\Users\{你的帳號名}\AppData\Local\Microsoft\Windows\INetCookies cookies文件格式請查看相關資料 下麵是清除cookies的代 ...
  • using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace _011 { //枚舉為值類型,不是引用類型 enum en_um { 畢畢, 男人, a = 1, b = ...
  • C 介面的顯示實現和隱示實現 public interface IReview { void GetReviews(); } public class ShopReview :IReview { public void GetReviews(){} } 這種方式是隱示實現: IReview rv = ...
  • 事件的定義過程 定義類型來容納所有需要發送給事件通知接收者的附加信息 簡單的說,就是調用事件的時候能夠傳遞一些參數信息 定義事件成員 事件成員類型為 ,意味著‘事件通知’的所有接收者都必須提供一個原型和 委托類型相匹配的回調方法, 所以方法原型必須具有以下形式 定義負責引發事件的方法來通知事件的登記 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...