1. 程式人生 > >C#中的Explicit和Implicit

C#中的Explicit和Implicit

編譯錯誤 rst itl msdn code 異常 public view 顯式

今天在Review一個老項目的時候,看到一段奇怪的代碼。

if (dto.Payment == null) continue;

var entity = entries.FirstOrDefault(e => e.LedgerEntryID == dto.LedgerEntryID);

dto.Payment = entity?.Payment;

 

其中dto.Payment是一個PaymentDTO類的實例,entity?.Payment是一個Payment類的實例,PaymentDTO類和Payment類沒有子父關系,所以不存在子類和父類之間的隱式轉換。

奇怪的是Visual Studio的編譯器沒有提示任何編譯錯誤。

打開PaymentDTO類的定義之後,發現了以下方法簽名。

    public static implicit operator PaymentDTO(Payment payment)

從方法簽名上看,這就是重寫PaymentDTO類型的操作符,但並不是我以前常用的+,-,*,/, ==等。

查詢MSDN之後,才了解到implicit和explicit是一對轉換操作符。

Implicit和Explicit

Implicit

Implicit關鍵字用於聲明隱式的用戶定義類型轉換運算符。它可以實現2個不同類的隱式轉換 ,提高代碼的可讀性。但是需要註意使用隱式轉換操作符之後,在編譯時會跳過異常檢查,所以隱式轉換運算符應當從不引發異常並且從不丟失信息,否則在運行時會出現一些意想不到的問題。

例如當前PaymentDTO和Payment的定義如下

    public class Payment

    {

         public decimal Amount { get; set; }

    }

 

    public class PaymentDTO

    {

         public string AmountString { get; set; }

    }

如果需要將Payment隱式轉換成PaymentDTO, 僅需聲明PaymentDTO的隱式轉換運算符

        public
class PaymentDTO { public string AmountString { get; set; } public static implicit operator PaymentDTO(Payment payment) { return new PaymentDTO { AmountString = payment.Amount.ToString("C2") }; } }

調用時只需要直接賦值就可以

        class Program

        {

            static void Main(string[] args)

            {

                PaymentDTO dto = new Payment { Amount = 1 };

 

                Console.WriteLine(dto.AmountString);

                Console.Read();

            }

        }

Explicit

Explicit關鍵字聲明必須通過轉換來調用的用戶定義的類型轉換運算符。不同於隱式轉換,顯式轉換運算符必須通過轉換的方式來調用,如果缺少了顯式的轉換,在編譯時就會產生錯誤。

例如現在我們將前面PaymentDTO類中定義的轉換操作符從Implicit變為Explicit

        public class PaymentDTO

        {

            public string AmountString { get; set; }

 

            public static explicit operator PaymentDTO(Payment payment)

            {

                return new PaymentDTO

                {

                    AmountString = payment.Amount.ToString("C2")

                };

            }

        }

這時候由於Main方法中沒有顯式轉換,所以編譯器出錯,提示Cannot implicitly convert type ‘ExplicitImplicit.Payment‘ to ‘ExplicitImplicit.PaymentDTO‘. An explicit conversion exists (are you missing a cast?)

技術分享圖片

如果想要編譯器通過編譯, 只需要做一個顯示轉換即可

        class Program

        {

            static void Main(string[] args)

            {

                PaymentDTO dto = (PaymentDTO)new Payment { Amount = 1 };

 

                Console.WriteLine(dto.AmountString);

                Console.Read();

            }

        }

總結

  • Implicit提高了代碼的可讀性,但程序員需要自己保證轉換不引發異常且不丟失信息
  • Explicit可阻止編譯器靜默調用可能產生意外後果的轉換操作。
  • 前者更易於使用,後者能向閱讀代碼的每個人清楚地指示您要轉換類型

C#中的Explicit和Implicit