所以不存在子类和父类之间的隐式转换

时间:2022-03-13 07:09:29

今天在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可阻止编译器静默挪用可能孕育产生不测后果的转换操纵。