成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深入解析C#編程中的方法重載

開發 后端
方法重載是讓類以統一的方式處理不同類型數據的一種手段。在C#中,語法規定同一個類中兩個或兩個以上的方法可以用同一個名字,如果出現這種情況,那么該方法就被稱為重載方法.當一個重載方法被調用時,C#回根據調用該方法的參數自動調用具體的方法來執行.對于方法的使用者來講,這種技術是非常有用的。本文將對C#編程中的方法重載進行詳細介紹。

首先我們說一下什么是方法重載。在面對對象這樣的高級語言中都允許我們在一個類中定義多個方法名相同、方法間參數個數和參數順序不同的方法,對于參數個數不同或者參數列表不同的情況我們稱之為參數列表不同。需要注意的是這里沒有提到方法的返回值。也就是決定方法是否構成重載有以下幾個條件:

◆ 在同一個類中;

◆ 方法名相同;

◆ 參數列表不同。

例如下面的代碼:

public void Show()//(1)
        {
            Console.WriteLine("Nothing");
        }
        public void Show(int number)//(2)
        {
            Console.WriteLine(number);
        }
        /*
        public int Show(int number)//(3)
        {
            Console.WriteLine(number);
            return number % 5;
        }
        */

對于上面的代碼,(1)沒有參數,(2)使用了一個int類型的參數,(1)和(2)之間就構成了重載。(2)與(3)相比僅僅返回值不同,雖然重載不關心返回值的不同,但是在C#中不允許存在方法名和參數列表相同、返回值不同的方法,所以(2)和(3)不能同時存在于代碼中,(3)如果不注釋掉上面的代碼是沒有辦法通過編譯的。

上面我們僅僅討論了重載的一些基本常識,下面我們探討一下一些情況稍微復雜的重載情況。

首先我們看第一個版本:

using System;
using System.Collections.Generic;
using System.Text;
namespace OverrideDemo
{
    /// <summary>
    /// 說明:本實例用于講述重載的關系
    /// 作者:周公
    /// 日期:2008-09-04
    /// 首發地址:http://blog.csdn.net/zhoufoxcn
    /// </summary>
    class Program
    {
        static void Main(string[] args)
        {
            String s = null;
            Show(s);
            Object o = "123";
            Show(o);
        }
        static void Show(string s)
        {
            Console.WriteLine("String");
        }
        static void Show(Object o)
        {
            Console.WriteLine("Object");
        }
    }
}

大家猜猜這個程序的運行結果是什么?

以下是程序運行結果:

String
Object

對以上代碼進行分析,我們發現Show()方法有兩種形式,一種是string類型的參數,一種是object類型參數,在一個類中存在方法名相同、參數列表不同(參數個數或者參數類型不同)的現象我們稱之為overloading,即重載。不過這里的Show()方法的參數比較特殊,因為string類繼承于Object類,也就是Show()方法的參數存在一種繼承關系。從結果我們可以得出兩點結論:

(1)從String s = null;Show(s);最后調用的是static void Show(string s)這個方法我們可以得出,C#中方法調用是精確匹配的,也就是s是string類型,雖然string類型繼承自object類型,盡管static void Show(Object o)也滿足條件,但是方法聲明中static void Show(string s)這個聲明與s類型的最接近(因為s是string類型,與它最接近),所以執行static void Show(string s),而不執行static void Show(Object o)這個方法。

(2)從Object o = "123"; Show(o);最后調用的是static void Show(Object o)這個方法我們可以得出,C#中如果存在方法重載,會根據其refrence type(引用類型)來調用對象的方法,而不是根據instance type(實例類型)來調用。盡管”123”是string類型,但是它的refrence type是object類型的,所以會調用static void Show(Object o)這個方法而不是static void Show(string s)。

#p#

上面的Main()方法的IL代碼如下:

.method private hidebysig static void  Main(string[] args) cil managed
{
  .entrypoint
  // 代碼大小       24 (0x18)
  .maxstack  1
  .locals init ([0] string s,
           [1] object o)
  IL_0000:  nop
  IL_0001:  ldnull
  IL_0002:  stloc.0
  IL_0003:  ldloc.0
  IL_0004:  call       void OverrideDemo.Program::Show(string)
  IL_0009:  nop
  IL_000a:  ldstr      "123"
  IL_000f:  stloc.1
  IL_0010:  ldloc.1
  IL_0011:  call       void OverrideDemo.Program::Show(object)
  IL_0016:  nop
  IL_0017:  ret
} // end of method Program::Main

從上面的IL代碼我們可以看出對于string s=null;這句代碼在IL中表示為:ldnull。

再根據上面的結論,我們看下面的代碼:

using System;
public class Program
{   public static void Main()
    {
    Show(null);
    Show("");
    Show(1);
    }
    static void Show(Object o)
    {
    Console.WriteLine("Object");
    }
   
    static void Show(String s)
    {
    Console.WriteLine("String");
    }
   
}

猜猜上面的代碼執行結果會是怎樣的?

以下是程序運行結果:

String
String
Object

從上面的運行結果我們可以得出以下結論:

(1)從Show(null)最后調用的是static void Show(String s)方法我們更進一步可以說在C#中是方法調用盡量精確匹配的。盡管null我們可以理解為一個空object對象或者一個空字符串,但是在這里C#還是精確為派生類。這就像我們沒有錢,可以說沒有一分錢也可以說沒有500英鎊,但是沒有一分錢自然就沒有500億英鎊,所以我們跟別人說沒有錢的時候沒有必要說沒有500億英鎊一樣。在這里自然null就表示空字符串。所以Show(null)這個方法會調用static void Show(String s)這個方法。

這有點像下面的情況:

一次活動大會上,主持人說:“身高不到1.60m的請坐在1到3排,身高不到1.75m的請做到4到6排,其他的請隨便坐。”

上面的語句似乎有些邏輯方面的問題,應該說身高超過1.60m但是不到1.75m的請坐到4到6排。但是如果你面對著一群拿著槍的強盜,他說上面的話時,恰好你也在場并且你的身高是1.55m,你會坐到哪一排?你總不可能冒著挨一槍的危險去糾正他的邏輯錯誤吧?最好的辦法是坐到1到3排。因為無論怎么說你的身高是絕對滿足不到1.60m這個條件的(盡管你的身高也滿足強盜說的第二個條件,即身高不到1.75米,但是你肯定不會冒這個危險,從上下句的意思我們也能推斷出人家的意思就是身高在1.61m到1.74m之間的人坐4到6排)。

在上面的代碼中,你在運行環境的眼中就是一個持槍的強盜,雖然null可以理解為null類型的string或者null類型的object,但是它不能向你問清楚這個到底是null類型的string或者null類型的object,因為string是Object的派生類,所以它按照null類型的string來調用相應的方法了。

(2)從Show("")最后調用static void Show(String s)這個方法進一步證明了方法調用是盡量選擇參數最匹配的那個執行。因為Show("")相當于:string s = ""; Show(s);s的引用類型是string,所以會調用static void Show(String s)這個方法。

#p#

我們在這里可以假設一下:假如存在一個類A是String類的派生類(實際上string類是sealed的,也就是不可繼承的,所以我說了是假設),并且存在在上面的代碼改變如下:

using System;
public class Program
{   public static void Main()
    {
    Show(null);
    Show("");
    }
    static void Show(Object o)
    {
    Console.WriteLine("Object");
    }
   
    static void Show(String s)
    {
    Console.WriteLine("String");
    }
    static void Show(A a)//假設A是String的派生類,當然實際上String類并沒有派生類,這里僅僅是假設
    {
    Console.WriteLine("A");
    }
   
}

如果上面的假設成立,上面的代碼運行結果應該如下:

A
String

(3)為什么Show(1)會調用static void Show(Object o)這個方法呢?在這個類中與Show(1)最精確的方法重載應該是static void Show(int i)這種方法聲明,但是方法中沒有,因為int是繼承自ValueType類,所以如果沒有static void Show(int i)這種聲明,那么其次接近的聲明應該是static void Show(ValueType v)這種聲明,可惜方法中依然沒有,不過ValueType類繼承自Object類,所以比static void Show(ValueType v)還次一點的方法重載聲明應該是static void Show(Object o),而類中也確實存在這種聲明,所以會調用static void Show(Object o)這個方法。當然從int到Object這個過程中存在一次box,也就是裝箱(裝箱是從值類型到引用類型的轉換),這個可以從下面的IL代碼可以看出來。

以下是第二種情況下Main()方法的IL代碼:

.method private hidebysig static void  Main(string[] args) cil managed
{
  .entrypoint
  // 代碼大小       32 (0x20)
  .maxstack  8
  IL_0000:  nop
  IL_0001:  ldnull
  IL_0002:  call       void OverrideDemo.Program::Show(string)
  IL_0007:  nop
  IL_0008:  ldstr      ""
  IL_000d:  call       void OverrideDemo.Program::Show(string)
  IL_0012:  nop
  IL_0013:  ldc.i4.1
  IL_0014:  box        [mscorlib]System.Int32
  IL_0019:  call       void OverrideDemo.Program::Show(object)
  IL_001e:  nop
  IL_001f:  ret
} // end of method Program::Main

下面我們對第二種情況的代碼做一些變化,代碼如下:

using System;
using System.Collections.Generic;
using System.Text;
namespace OverrideDemo
{
    /// <summary>
    /// 說明:本實例用于講述重載的關系
    /// 作者:周公
    /// 日期:2008-09-04
    /// 首發地址:http://blog.csdn.net/zhoufoxcn
    /// </summary>
    class Program
    {
        static void Main(string[] args)
        {
            Show(null);
            Show(""); 
            Show(1);
        }
        static void Show(string s)
        {
            Console.WriteLine("String");
        }
        static void Show(Object o)
        {
            Console.WriteLine("Object");
        }
        static void Show(Program p)//Program是當前方法所在的類
        {
            Console.WriteLine("Program");
        }
    }
}

上面的代碼的運行結果是什么,你能猜出來嗎?

哈哈,上面的程序代碼是沒有運行結果的,因為它沒有辦法編譯!

為什么不能通過編譯呢?

原因就出在Show(null)這個方法這里!如果僅僅有static void Show(string s)和static void Show(Object o)方法構成重載關系,那么null我們既可以理解為空string引用也可以理解為空Object引用,因為string類型的限制更精確一些,所以C#會按照最精確地匹配成string類型,因而會執行static void Show(string s)這個方法。這是在前面的代碼中已經被證明的。可是現在多了一個static void Show(Program p)方法的重載,null既可以理解成空string類型引用,也可以理解成空Program類型引用,因為string類和Program類都是Object類的派生類,所以按照前面的推論自然不會當成空Object類型的引用。因為String類和Program類之間不存在繼承關系,按照最精確匹配原則,編譯器無法決定匹配成String類還是Program類最精確,所以編譯無法通過。

【編輯推薦】

  1. C#連接數據庫特殊方法兩則
  2. 深入理解C# 3.0的五項主要改進
  3. C#中的Namespace詳解
責任編輯:楊鵬飛 來源: zhoufoxcn的博客
相關推薦

2009-08-19 14:42:21

C#重載LoadPos

2009-08-12 10:56:47

C#運算符重載C#運算符重載實例

2009-08-25 17:15:50

C#隱藏C#重寫C#重載

2024-05-06 00:00:00

C#工具代碼

2009-08-31 18:17:32

C#接口編程

2024-05-11 07:13:33

C#Task編程

2024-06-25 08:33:48

2024-06-24 08:36:54

2009-04-10 09:55:44

C#反射.NET

2009-08-05 14:09:04

C#日期轉換

2024-05-15 08:09:23

2009-07-31 16:00:30

C#函數重載

2024-09-18 05:20:00

C#代碼

2024-05-06 00:00:00

ThreadPool線程調度

2009-03-27 09:26:20

定義Duck TypingC#

2009-08-21 09:44:44

C#發送Email郵件

2024-01-03 13:38:00

C++面向對象編程OOP

2009-09-01 11:19:47

C# 3.0擴展重載抉

2009-09-09 14:04:18

C# XML解析XML解析方法

2009-08-28 12:31:06

C#靜態方法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文字幕免费视频 | 国产在线一区观看 | 日本不卡免费新一二三区 | 久久久久国产一区二区三区四区 | 欧美一区二区三区 | 在线观看视频91 | 福利片一区二区 | 国产免费观看久久黄av片涩av | 日本公妇乱淫xxxⅹ 国产在线不卡 | 一级免费看片 | 91精品国产综合久久久久久丝袜 | 一级欧美 | 国产精品毛片一区二区在线看 | 久草视| 久一精品 | 不卡一区| 日韩精品在线网站 | 国产精品美女久久久久久免费 | 国产精品久久久久aaaa | 亚洲 欧美 另类 综合 偷拍 | 激情一区二区三区 | 国产一区中文 | 久久不射网| 久久草在线视频 | 午夜视频一区二区三区 | 成人一区二区三区在线观看 | 国产精品精品视频一区二区三区 | 99精品久久久 | 日韩在线日韩 | 久久成人一区 | 欧洲一区二区三区 | 国产成人精品午夜视频免费 | 免费美女网站 | 日日操操操 | 97国产一区二区精品久久呦 | yiren22 亚洲综合 | a级片在线观看 | 午夜精品 | 91精品国产综合久久香蕉922 | 五十女人一级毛片 | 黄瓜av |