C#不用union,而是有更好的方式實現

来源:https://www.cnblogs.com/hjsstudio/archive/2018/05/24/9082218.html
-Advertisement-
Play Games

用過C/C++的人都知道有個union,特別好用,似乎char數組到short,int,float等的轉換無所不能,也確實是能,並且用起來十分方便。那C#為什麼沒有這個關鍵字呢?怎麼實現這個功能?其實C#只是沒有了這個關鍵字,但是功能是能實現的,而且也是非常方便,並且是安全的。網上有人用Struct ...



用過C/C++的人都知道有個union,特別好用,似乎char數組到short,int,float等的轉換無所不能,也確實是能,並且用起來十分方便。
那C#為什麼沒有這個關鍵字呢?怎麼實現這個功能?其實C#只是沒有了這個關鍵字,但是功能是能實現的,而且也是非常方便,並且是安全的。
網上有人用StructLayout特性來實現union,也確實是實現了一些功能。
比如:
C/C++:
    union {
        unsigned char ch
        short ;
        int i;
    };
C#:
    [StructLayout(LayoutKind.Explicit)]
    public struct Class1
    {
        [FieldOffset(0)]
        public byte b;

        [FieldOffset(0)]
        public short s;

        [FieldOffset(0)]
        public int i;
    }
就可以實現。
但是我要是寫個:
    union {
        unsigned char ch[4];
        int i;
        float f;
    } temp;
硬是用C#沒有模擬出來,估計我還沒有找著合適的方法。因為我寫
    [StructLayout(LayoutKind.Explicit)]
    public struct Class1
    {
        [FieldOffset(0)]
        public byte[4] b;

        [FieldOffset(0)]
        public short s;

        [FieldOffset(0)]
        public int i;
    }
這玩意是編譯不通過的。然後折騰了半天,沒有折騰出來。後來又回到C/C++想了一番,似乎有些認識。
C/C++用union其實就是使用同一塊記憶體存儲不同類型的數據,說白了,就是一塊公用的記憶體,你用啥讀取出來就是啥內容。其實電腦中的記憶體本身也就是這樣,你定義一個int i;然後電腦會在記憶體棧上開闢一塊空間,並且這塊記憶體指明瞭是int類型,但是我們經常看到(int)data,(int*)pt等操作,說明可以強制轉換。強制轉換不是說把這幾塊記憶體的值改變了,只是臨時改變了讀取方式,然後用這種方式讀取這塊記憶體。那這樣說來是不是也可以不用union來實現char數組與其他類型之間的轉換,答案是必須可以。
比如:
    unsigned char chArr[4] = "";
    float f1 = 45.56f;
    memcpy(chArr, &f1, sizeof(float));
    // 運行結果:113    61    54    66
    printf("%d\t%d\t%d\t%d\n", chArr[0], chArr[1], chArr[2], chArr[3]);
    
    float f2 = 0.00f;
    memcpy(&f2, chArr, sizeof(float));
    printf("%0.2f\n", f2);
    
    float f3 = *(float *)chArr;
    printf("%0.2f\n", f3);

    char *pch = (char *)&f3;
    // 運行結果:113    61        54        66
    printf("%d\t%d\t%d\t%d\n", pch[0], pch[1], pch[2], pch[3]);

那好問題來了,C#怎麼實現?
那好,答案也來了。當然是用BitConvert。
比如:
    float f = 45.56f;
    byte[] b = BitConverter.GetBytes(f);
    Console.WriteLine("bArr\t: {0}\t{1}\t{2}\t{3}", b[0], b[1], b[2], b[3]);

    float f2 = BitConverter.ToSingle(b, 0);
    Console.WriteLine("f2\t: {0}", f2);
完全木有問題啊,而且還安全。

最後呢,咱們看看微軟是怎麼給咱實現的。

    // Converts a float into an array of bytes with length
    // four.
    [System.Security.SecuritySafeCritical]  // auto-generated
    public unsafe static byte[] GetBytes(float value)
    {
        Contract.Ensures(Contract.Result<byte[]>() != null);
        Contract.Ensures(Contract.Result<byte[]>().Length == 4);

        return GetBytes(*(int*)&value);
    }

    ...
    // Converts an int into an array of bytes with length
    // four.
    [System.Security.SecuritySafeCritical]  // auto-generated
    public unsafe static byte[] GetBytes(int value)
    {
        Contract.Ensures(Contract.Result<byte[]>() != null);
        Contract.Ensures(Contract.Result<byte[]>().Length == 4);

        byte[] bytes = new byte[4];
        fixed(byte* b = bytes)
            *((int*)b) = value;
        return bytes;
    }

看見了嗎?是不是跟上面的C/C++代碼很像。其實就是C/C++代碼。如果你看不到這段代碼,也許你還真不知道,原來以前自己的C/C++代碼被搬到了這裡。但是微軟的公司的代碼可不是我寫的C/C++那麼簡單的轉換,微軟程式員是做了安全檢查的。你如果將3個byte的數組轉換到float,那對不起,玩不了,你得補一個位元組。

好了,給大家附上微軟C#開源的源代碼地址:
https://referencesource.microsoft.com/#mscorlib/system/bitconverter.cs,9108fa2d0b37805b



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

-Advertisement-
Play Games
更多相關文章
  • Thread (ParameterizedThreadStart) 初始化 Thread 類的新實例,指定允許對象線上程啟動時傳遞給線程的委托。 Thread (ThreadStart) 初始化 Thread 類的新實例。 由 .NET Compact Framework 支持。 Thread (P ...
  • 先上結果: 之前 在公司業務中用過java+Selenium+ChromeDriver ,使用起來非常順手,可以完美模擬真實的用戶瀏覽行為。最近休息的時候想用C#也試一下,於是有了本文。 實現原理一樣,只是由java換成了C#。(ps:個人感覺就業務開發代碼來說,熟悉之後兩種語言可以無縫切換。) 事 ...
  • 百度,一家讓人既愛又恨的企業,血友吧貼吧被賣,魏則西事件的持續發酵,一時間將百度推到了輿論的風口浪尖上。是非對錯,我們在這裡也不多做評判,本文呢為大家整理了百度開源的70+項目,看看有沒有感興趣的。本文內容綜合整理自oschina、github。 1. JavaScript圖表庫 ECharts E ...
  • 為了減少由於單個請求掛掉而拖垮整站的情況發生,給所有請求做統計是一個不錯的解決方法,通過觀察哪些請求的耗時比較長,我們就可以找到對應的介面、代碼、數據表,做有針對性的優化可以提高效率。在 asp.net web api 中我們可以通過註冊一個 DelegatingHandler 來實現該功能。那在  ...
  • 1、什麼是編程語言? 程式員與電腦溝通的介質 2、什麼是編程? 程式員基於某種編程語言的語法格式將想讓電腦所做的事寫到文件中讓電腦執行,編程的結果就是文件,文件的內容就是程式; 3、為什麼要編程? 讓電腦代替人類工作,解放人力 4,、電腦硬體基礎 (1)、什麼是x86-64位? x86是c ...
  • 正確開啟虛擬化的方式 列表如ListBox,ListView,TreeView,GridView等,開啟虛擬化 直接在模板中,設置CanContentScroll="True" 如模板中未設置CanContentScroll屬性,可以在列表添加屬性ScrollViewer.CanContentScr ...
  • 回到目錄 Invoke和BeginInvoke都是調用委托實體的方法,前者是同步調用,即它運行在主線程上,當Invode處理時間長時,會出現阻塞的情況,而BeginInvod是非同步操作,它會從新開啟一個線程,所以不會租塞主線程,在使用BeginInvoke時,如果希望等待執行的結果 ,可以使用End ...
  • .Net Framework中,把資源分為托管資源和非托管資源兩大類, 托管資源指可以通過.Net Frame垃圾回收器進行回收的資源,主要是指分配在托管堆上你的記憶體資源,這類資源的回收是不需要人工干預,.Net Framework的垃圾回收器會在合適的時刻進行回收,程式也可以主動調用GC.Coll ...
一周排行
    -Advertisement-
    Play Games
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...