Selecione dinamicamente as informações JSON necessárias
-
20-12-2019 - |
Pergunta
Preciso desserializar algum JSON, mas o problema é que todos os meus objetos JSON não têm exatamente o mesmo formato.Aqui está um exemplo do JSON que preciso desserializar:
[
{
"Player_Name": "Zlatan Ibrahimovic",
"Country": "Sweeden",
"Other_Informations": {
"Information1": [
]
},
},
{
"Player_Name": "Pavel Nedved",
"Country": "Czech Republic",
"Personal_Honours":
{
"Ballon_DOr": "One",
},
"Other_Informations": {
"Information1": [
]
},
},
{
"Player_Name": "Zinedine Zidane",
"Country": "Sweeden",
"Other_Informations": {
"Information1": [
{
"name": "example",
}
]
},
}
]
Como você pode ver, alguns campos aparecem apenas para alguns objetos, por exemplo, "Personal_Honors".Preciso desserializar o JSON nesta classe:
public class PlayerData
{
public string Name { get; set; }
public string BallonDor {get; set; }
public string Information1{ get; set; }
public string Country{ get; set; }
}
Eu uso este método que é muito longo e bloqueia meu aplicativo:(Neste exemplo eu uso Json que vem de um arquivo de texto, mas normalmente tenho que fazer uma solicitação REST.)
StreamReader reader = File.OpenText("TextFile1.txt");
List<PlayerData> DataList;
dynamic value= JsonConvert
.DeserializeObject<dynamic>(reader.ReadToEnd());
DataList = new List<PlayerData>();
foreach (dynamic data in value)
{
if (data.Personal_Honours == null)
{
if (data.Other_Informations.Information1 == null)
{
DataList.Add(new PlayerData
{
Name = data.Player_Name,
Country = data.Country,
});
}
else
{
DataList.Add(new PlayerData
{
Name = data.Player_Name,
Country = data.Country,
Information1 = data.Informations.Information1
});
}
}
else
{
if (data.Other_Informations.Information1 == null)
{
DataList.Add(new PlayerData
{
Name = data.Player_Name,
Country = data.Country,
BallonDor = data.Personal_Honours.Ballon_DOr
});
}
else
{
DataList.Add(new PlayerData
{
Name = data.Player_Name,
Country = data.Country,
BallonDor = data.Personal_Honours.Ballon_DOr,
Information1 = data.Informations.Information1
});
}
}
}
Este método está funcionando, mas não é eficiente e bloqueia minha IU.Como posso criar um novo objeto "PlayerData" sem ter todas essas instruções 'else if'?
Obrigado !
P.S:A pergunta é diferente desta Filtrar informações JSON
EDITAR:
Aqui está como consegui o RunTimeBinderException :
List<PlayerData> datalist = new List<PlayerData>();
foreach (dynamic pl in timeline)
{
datalist.Add(new PlayerData
{
Name = pl.Player_Name ,
Country = pl.Country ,
BallonDor = pl.Personal_Honours.Ballon_Dor,
Information1 = pl.Other_Informations.Information1.name
});
}
Solução
Você recebeu uma exceção porque alguns dados não possuem Personal_Honours
propriedade por exemplo.Então você tentou acessar Ballon_Dor
propriedade de um null
referência que aciona a exceção.Desta forma funcionará para o exemplo JSON que você postou:
List<PlayerData> datalist = new List<PlayerData>();
foreach (dynamic pl in timeline)
{
//object initialization is done later just for the sake of easier debugging
//...so we can spot unexpected data which cause exception easily
var Name = pl.Player_Name;
var Country = pl.Country;
var BallonDor = pl.Personal_Honours != null ? pl.Personal_Honours.Ballon_Dor : null;
var Information1 = pl.Other_Informations.Information1.Count > 0 ?
pl.Other_Informations.Information1[0].name :
null;
datalist.Add(new PlayerData
{
Name = Name ,
Country = Country ,
BallonDor = BallonDor,
Information1 = Information1
});
}
... mas a abordagem acima ainda está sujeita a erros, dependendo de quão consistente é a string JSON que temos.Uma abordagem mais robusta talvez seja ter classes de modelo para mapear a string JSON, conforme sugerido por @L.B no comentário.