Eu mudei para o C # 8 em um dos meus projetos. E eu tenho movido todas as minhas switch
declarações para expressões. No entanto, descobri que meu projeto começou a funcionar de maneira diferente e descobri que era por causa da switch
expressão. Vamos obter esse código, por exemplo
class Program
{
public enum DataType
{
Single,
Double,
UInt16,
UInt32,
UInt64,
Int16,
Int32,
Int64,
Byte
}
static void Main(string[] args)
{
dynamic value1 = 5;
dynamic value2 = 6;
var casted = CastToType(value1, DataType.Int16);
var casted1 = CastToTypeExpression(value2, DataType.Int16);
var type = casted.GetType(); // Int16
var type1 = casted1.GetType(); // Double
var bytes = BitConverter.GetBytes(casted); // byte arr with 2 el => [5, 0] <- expected behavior
var bytes1 = BitConverter.GetBytes(casted1); // byte arr with 8 el => [0, 0, 0, 0, 0, 0, 24, 64]
}
public static dynamic CastToType(dynamic value, DataType type)
{
switch (type)
{
case DataType.Byte:
return (byte)value;
case DataType.Double:
return (double)value;
case DataType.Int16:
return (short)value;
case DataType.Int32:
return (int)value;
case DataType.Int64:
return (long)value;
case DataType.Single:
return (float)value;
case DataType.UInt16:
return (ushort)value;
case DataType.UInt32:
return (uint)value;
case DataType.UInt64:
return (ulong)value;
default: throw new InvalidCastException();
}
}
public static dynamic CastToTypeExpression(dynamic value, DataType type)
{
return type switch
{
DataType.Byte => (byte)value,
DataType.Double => (double)value,
DataType.Int16 => (short)value,
DataType.Int32 => (int)value,
DataType.Int64 => (long)value,
DataType.Single => (float)value,
DataType.UInt16 => (ushort)value,
DataType.UInt32 => (uint)value,
DataType.UInt64 => (ulong)value,
_ => throw new InvalidCastException(),
};
}
}
Eu escrevi o resultado como um comentário, mas tl; dr quando o switch clássico é usado lançando o valor retorna o valor no Type esperado, mas quando a expressão do switch é usada, ele retorna o tipo de "Double", resultando em diferente byte[]
ao obter o bytes do valor.
Qual a diferença entre os dois? Do que sinto falta?
Respostas:
No formulário de declaração do switch , cada braço está retornando um valor diretamente. Ele está convertendo do tipo numérico diretamente para
object
, pois esse é efetivamente o tipo de retorno do método.Seu formulário de expressão da opção é um pouco diferente. Primeiro extrai um resultado da expressão do comutador e depois converte esse resultado no tipo de retorno declarado. Então, qual é o tipo da expressão switch? É o tipo "melhor" de todos os tipos de expressões individuais nos braços da expressão switch.
Todos esses tipos podem ser implicitamente convertidos em
double
(que é um dos tipos em si), então esse é o melhor tipo. Portanto, seu método de expressão de chave é equivalente a:Você pode ver esse "melhor tipo" sem usar uma expressão de opção, usando matrizes de tipo implícito:
Aqui o tipo de
array
é inferido como sendodouble[]
.fonte