歡迎光臨
每天分享高質量文章

C# 8.0 中開啟預設介面實現

當你升級到 C# 8.0 和 .NET Core 3.0 之後,你就可以開始使用預設介面實現的功能了。

從現在開始,你可以在介面裡面新增一些預設實現的成員,避免在介面中新增成員導致大量對此介面的實現崩潰。

 

要寫出並且正常使用介面的預設實現,你需要:

  • C# 8.0
  • .NET Core 3.0
  • Visual Studio 2019 Preview (16.1 以上版本)
  • 前往下載安裝 Visual Studio Preview

對於預覽版的 Visual Studio 2019 來說,.NET Core 的預覽版是預設開啟且無法關閉的,所以不需要關心。

請設定你專案的屬性,修改 C# 語言版本為 8.0(對於預覽版的語言來說,這是必要的):

或者直接修改你的專案檔案,加上 LangVersion 屬性的設定,設定為 8.0。


  
    Exe
    netcoreapp3.0
    8.0
  


 

比如,我們現在有下麵這樣一個簡單的介面:

public interface IWalterlv
{
    void Print(string text);
}

這個介面被大量實現了。

現在,我們需要在介面中新增一個方法 DouBPrint,其作用是對 Print 方法進行標準化,避免各種不同實現帶來的標準差異。於是我們新增一個方法:

    public interface IWalterlv
    {
        void Print(string text);

++      void DouBPrint(string text);
    }

然而我們都知道,這樣的修改是破壞性的:

  1. 會使得所有實現這個介面的程式碼全部失敗(無法編譯透過,或者執行時丟擲異常)
  2. 我們依然很難將介面的實現標準化,靠檔案來規約

那麼現在,我們可以這樣來新增此方法:

    public interface IWalterlv
    {
        void Print(string text);
        
--      void DouBPrint(string text);
++      public void DouBPrint(string text) => Print($"Walterlv 逗比 {text}");
    }

在使用此方法來定義此介面中的方法後,那些沒來得及實現此方法的型別也可以編譯透過並獲得標準化的實現。

class Program
{
    static void Main(string[] args)
    {
        IWalterlv walterlv = new Foo();
        walterlv.DouBPrint("walterlv");
    }
}

public class Foo : IWalterlv
{
    public void Print(string text)
    {
    }
}

當然,對於 Foo 型別來說,實現也是可以的:

public class Foo : IWalterlv
{
    public void Print(string text)
    {
    }

    public void DouBPrint(string text) => Print($"Walterlv 逗比 {text}");
}

除此之外,在介面中還可以編寫靜態欄位和靜態方法,這可以用來統一介面中的一些預設實現。

意味著,如果類沒有實現介面中帶有預設實現的方法,那麼具有預設的實現;而如果類中打算實現介面中的帶有預設實現的方法,那麼也可以呼叫介面中的靜態方法來進行實現。

    public interface IWalterlv
    {
        void Print(string text);

--      public void DouBPrint(string text) => Print($"Walterlv 逗比 {text}");
++      public void DouBPrint(string text) => DefaultDouBPrint(this, text);
++
++      private static readonly string _name = "walterlv";
++
++      protected static void DefaultDouBPrint(IWalterlv walterlv, string text)
++          => walterlv.Print($"{_name} 逗比 {text}");
    }

然後,對於實現方,則需要使用介面名來呼叫介面中的靜態成員:

    public class Foo : IWalterlv
    {
        public void Print(string text)
        {
        }

--      public void DouBPrint(string text) => Print($"Walterlv 逗比 {text}");
++      public void DouBPrint(string text)
++      {
++          // Do Other things.
++          IWalterlv.DefaultDouBPrint(this, text);
++      }
++  }

參考資料

  • Default implementations in interfaces – .NET Blog
  • Visual Studio 2019 version 16.1 Preview 3 – The Visual Studio Blog
  • Safely update interfaces using default interface members in C# – Microsoft Docs
贊(0)

分享創造快樂