C#中的Explicit和Implicit
今天在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