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

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)

分享創造快樂