理解靜態成員(static)的本質---類型對象

来源:https://www.cnblogs.com/marsir/archive/2018/04/12/8811233.html
-Advertisement-
Play Games

我們只知道靜態成員的用法,比如調用一個靜態方法要用類名去點出來,比如Math.Max(),但是為什麼是這樣調用呢?為什麼靜態構造函數不能有訪問修飾符?為什麼靜態構造函數沒有參數?靜態函數的執行時間又是什麼時候?為什麼? 在面向對象中,萬物皆對象。其中就有一種對象叫作類型對象,類型對象就是我們創建一個 ...


我們只知道靜態成員的用法,比如調用一個靜態方法要用類名去點出來,比如Math.Max(),但是為什麼是這樣調用呢?為什麼靜態構造函數不能有訪問修飾符?為什麼靜態構造函數沒有參數?靜態函數的執行時間又是什麼時候?為什麼?

在面向對象中,萬物皆對象。其中就有一種對象叫作類型對象,類型對象就是我們創建一個class時所代表的對象。

    class Program
    {
        public static string name;//靜態欄位 
        public string tag;//實例欄位 
        static void Main(string[] args)
        {
            Program p = new Program();//Program是類型對象,p是這個類型對象的實例對象
        }
    }

在C#中,每個AppDomain都保證了一個類型對象的唯一性(原理後面再解釋)。

類型對象是實例對象的模板,具體來說,就是運用new操作符實例化一個對象的時候,CLR首先會計算類型對象中定義的實例欄位所需要的位元組數,然後會為對象分配兩個額外的欄位(類型對象指針和同步塊索引)所需要的位元組(順便說下,C#中每個對象都有兩個額外的欄位),最後返回對象的引用。其中實例對象的類型對象指針指向的就是類型對象。

既然說了所有對象都有兩個額外的欄位,那麼類型對象顯然也是有的,類型對象指向的是Type類型,即所有的類型對象都相當於是Type類型對象的實例。而Type類型的類型對象指針指向的是它本身。這也就理解了下麵這個代碼的意義。

 Type t1 = p.GetType();
 Type t2 = typeof(Program);

同時也可以利用類型對象指針去思考下虛方法的調用原理。

那麼類型對象什麼時候初始化呢?它又是用什麼來初始化呢?

要創建一個類型的實例或者調用類型對象的成員的時候,才需要初始化類型對象。因為畢竟只有有了模塊才能實例化出來,有了對象才能使用它的成員,所有的靜態成員都是屬於類型對象而不屬於類型的實例,所以調用的時候用的是類名直接點出來。就像類型的實例方法,是用類型的實例名點出來一樣。在C#中,對象的初始化幾乎都是通過調用構造函數來實現的(object的MemberwiseClone,以及反序列化除外)。要初始化一個類型對象必須調用類型對象的構造函數,它就是我們所知道的靜態構造函數(類型構造器)。類型的靜態構造函數是用來初始化類型對象的,而實例構造函數是用來初始化類型的實例對象的。

程式運行時,我們無法創建類型對象,類型對象的創建都是由CLR來執行的,所以靜態構造函數是沒有訪問修飾符的,即它永遠是private。因此它也沒有參數,即使有,也沒有什麼意義,因為都是由CLR執行的,所以類型構造器是沒有參數且不能有訪問修飾符。在程式集的清單元數據文件中,記錄了若幹個表,其中就有方法定義表,大致就是記錄方法的名稱、標誌、索引等。在CLR編譯一個方法的時候,它會從元數據中看下這個方法中都引用了哪些類型,然後檢查當前的AppDomain中是否已經執行了這個類型的構造器,如果沒有執行,那麼就執行它。當然方法可能被多個線程同時訪問,所以調用靜態構造函數的時候,調用線程會獲得一個互斥鎖,其它線程等待,當調用線程執行完成以後就會,其它線程再次進入方法時發現方法已經被執行了就會直接返回。

這就是類型對象。

其實泛型類也是類型對象。而且由泛型類型創建出的類型對象也是分別獨立的。

namespace Static
{
    class Program
    {
        static void Main(string[] args)
        {
            GenericType<int>.name = "Close Type";
           // Console.WriteLine(GenericType<>.name);               //C#不支持開放類型,所以會報錯
            Console.WriteLine(GenericType<int>.name);              //列印結果 : Close Type
            Console.WriteLine(GenericType<float>.name);            //列印結果 : Open Type
            Console.ReadKey();
        }
    }
    class GenericType<T>                                           //GenericType<>類型對象
    {
        public static string name = "Open Type";
    }
}

 可以看到GenericType<int>、GenericType<float>、GenericType<>是不同的類型對象。所以現在更好理解const、readonly、static readonly之間的區別了。


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

-Advertisement-
Play Games
更多相關文章
  • * 一個邏輯表達式里有多種運算符時,計算順序為: (判斷大小的)表達式 > and > or * content = input('xxx') 時, content的數據類型是str類型, 無論輸入的是什麼 * while的用法: 基本用法: 簡寫用法1: 簡寫用法2: flag用法: flag簡寫 ...
  • C++的 bitset 在 bitset 頭文件中,它是一種類似數組的結構,它的每一個元素只能是0或1,每個元素僅用1bit空間。 下麵是具體用法 構造函數 bitset常用構造函數有四種,如下 註意: 用字元串構造時,字元串只能包含 '0' 或 '1' ,否則會拋出異常。 構造時,需在<>中表明b ...
  • 寫入文本文件 1. 關聯讀入的文件,使用Reader 和 FileReader 2. 關聯寫出的文件,使用Writer和 FileWriter 3. 創建緩衝 char數組,用於接收讀取到的文本信息 4. 將文本讀入到 緩衝數組(buff)中 5. 輸出讀取到的文本信息 6. 寫出讀取到的文件 7. ...
  • 練習 4.12: 流行的web漫畫服務xkcd也提供了JSON介面。例如,一個 https://xkcd.com/571/info.0.json 請求將返回一個很多人喜愛的571編號的詳細描述。 下載每個鏈接(只下載一次)然後創建一個離線索引。編寫一個xkcd工具,使用這些離線索引,列印和命令行輸入 ...
  • 在Kotlin中 使用js 函數 ...
  • 在一對多或者多對多的時候。如果通過一的一方取獲得多的一方的數據。除了第一次查詢表的數據以外。每獲得一條多的一方的數據就查詢一次。 如:通過學生表的記錄查詢成績表的記錄。 一個學生就查詢一次,50個學生就查詢50次。 如果需要查詢50個學生的成績,需要查詢資料庫的次數為 第一次查詢學生的記錄+50次查 ...
  • 概述 UWP Community Toolkit Extensions 中有一個為可視元素提供的擴展 - VisualExtensions,本篇我們結合代碼詳細講解 VisualExtensions 的實現。 VisualExtensions 為可視元素提供了一種簡單的在 XAML 中修改通用屬性的 ...
  • 搭建Spring: 3、演示IOC操作,在Service中調用Dao中的方法 UserDao.java package com.zzb.www.dao public class UserDao{ public void add(){ System.out.println("dao........") ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...