0

如何反序列化在单个类中有块的 JSON 文件?有没有一种方法可以使用 Class 中的 JSON 注释来告知属性的父块和属性的父块是什么?

JSON如下:

{
    "Viagem": {
        "Id": 33333,
        "NumeroAtracacao": "22/2222",
        "NumeroViagem": "02002 00303",
        "Status": "DESATRACADO",
        "Joint": "UCLA UCLA",
        "Servico": "AMERICA CENTRAL",
        "MotivoEspera": "-",
        "LiberacaoRecebimento": "21/05/2018 07:00:00",
        "Navio": {
            "Nome": "MONTE CERVANTES",
            "Armador": {
                "Id": 0,
                "CodigoGeParcei": null,
                "Nome": "ALIANCA",
                "Sigla": "ALI",
                "CnpjCpf": null,
                "Endereco": null,
                "Cep": null,
                "Site": null
            },
            "ImagemNavio": ".......",
            "Comprimento": 272.08,
            "Lloyd": 9283186,
            "CallSign": "DHTK",
            "CapacidadeTeus": 5560,
            "Shortname": "MOCER"
        },
        "ChegadaPrevista": "27/05/2018 12:00:00",
        "AtracacaoPrevista": "29/05/2018 07:00:00",
        "SaidaPrevista": "30/05/2018 19:00:00",
        "DeadLine": "25/05/2018 12:00:00"
    }
}

这是我想用来反序列化 JSON 的类:

namespace WS_SantosBrasil.Model
{
    using System;
    using System.Collections.Generic;
    using System.ComponentModel.DataAnnotations;
    using System.ComponentModel.DataAnnotations.Schema;
    using System.Data.Entity.Spatial;

    [Table("Viagem")]
    public partial class Viagem
    {
        [Key]
        [Column(Order = 0)]
        [DatabaseGenerated(DatabaseGeneratedOption.None)]
        public int Id { get; set; }  

        [Key]
        [Column(Order = 1)]
        [StringLength(50)]
        public string IdCtis { get; set; }

        [StringLength(20)]
        public string NumeroAtracacao { get; set; }

        [StringLength(20)]
        public string NumeroViagem { get; set; }

        [StringLength(50)]
        public string Status { get; set; }

        [StringLength(50)]
        public string Joint { get; set; }

        [StringLength(50)]
        public string Servico { get; set; }

        [StringLength(50)]
        public string MotivoEspera { get; set; }

        [StringLength(20)]
        public string LiberacaoRecebimento { get; set; }

        [StringLength(20)]
        public string ChegadaPrevista { get; set; }

        [StringLength(20)]
        public string AtracacaoPrevista { get; set; }

        [StringLength(20)]
        public string SaidaPrevista { get; set; }

        [StringLength(20)]
        public string DeadLine { get; set; }

        [StringLength(20)]
        public string Chegada { get; set; }

        [StringLength(20)]
        public string Atracacao { get; set; }

        [StringLength(20)]
        public string Saida { get; set; }

        [StringLength(20)]
        public string InicioOperacao { get; set; }

        [StringLength(20)]
        public string FimOperacao { get; set; }

        [StringLength(50)]
        public string TipoOperacao { get; set; }

        [StringLength(20)]
        public string CodigoCodesp { get; set; }

        public decimal? CaladoAtracacao { get; set; }

        public decimal? CaladoDesatracacao { get; set; }

        [StringLength(20)]
        public string NumeroViagemImportacao { get; set; }

        [StringLength(20)]
        public string NumeroViagemExportacao { get; set; }

        [StringLength(20)]
        public string PrevisaoDescarga { get; set; }

        [StringLength(20)]
        public string PrevisaoEmbarque { get; set; }

        [StringLength(20)]
        public string PrevisaoRemocao { get; set; }

        [StringLength(50)]
        public string LocalAtracacao { get; set; }

        [StringLength(100)]
        public string Navio_Nome { get; set; }

        public string ImagemNavio { get; set; }

        public decimal? Navio_Comprimento { get; set; }

        [StringLength(20)]
        public string Navio_Lloyd { get; set; }

        [StringLength(50)]
        public string Navio_CallSign { get; set; }

        public int? Navio_CapacidadeTeus { get; set; }

        [StringLength(20)]
        public string Navio_Shortname { get; set; }

        public int? Armador_Id { get; set; }

        [StringLength(20)]
        public string Armador_CodigoGeParcei { get; set; }

        [StringLength(100)]
        public string Armador_Nome { get; set; }

        [StringLength(10)]
        public string Armador_Sigla { get; set; }

        [StringLength(20)]
        public string Armador_CnpjCpf { get; set; }

        [StringLength(100)]
        public string Armador_Endereco { get; set; }

        [StringLength(20)]
        public string Armador_Cep { get; set; }

        [StringLength(100)]
        public string Armador_Site { get; set; }

        public virtual ContainerViagem ContainerViagem { get; set; }
    }
}

例如,在反序列化此类的 JSON 时,如何通知属性 Nome_Navio 位于块 Navio 内,并且 JSON 属性为 NOME?

4

3 回答 3

2

你确定你不想像 CodeCaster 说的那样创建一个类来映射它吗?

public class Viagem 
{
    public int Id {get; set;}
    public string NumeroAtracacao {get; set;}
    public string Status {get; set;}
    public string Joint {get; set;}
    public string Servico {get; set;}
    public string MotivoEspera {get; set;}
    public Navio Navio {get; set;}
    public DateTime ChegadaPrevista {get; set;}
    public DateTime AtracacaoPrevista {get; set;}
    public DateTime SaidaPrevista {get; set;}
    public DateTime DeadLine {get; set;}
}

public class Navio 
{
    public string Nome {get; set;}
    public Armador Armador {get; set;}
    public string ImagemNavio {get; set;}
    public int Comprimento {get; set;}
    public int Lloyd {get; set;}
    public string CallSign {get; set;}
    public string CapacidadeTeus {get; set}
    public string Shortname {get; set;}
}

public class Armador 
{
    public int Id {get; set;}
    public object CodigoGeParcei {get; set;}
    public string Nome {get; set;}
    public string Sigla {get; set;}
    public object CnpjCpf {get; set;}
    public object Endereco {get; set;}
    public object Cep {get; set;}
    public object Site {get; set;}
}

然后你可以直接做

JsonConvert.DersializeObject<Viagem>(json);(如果您使用的是 Newtonsoft)

于 2018-07-26T13:33:11.043 回答
1

您应该能够使用Can I specify a path in an attribute 中的JsonPathConverter将我的类中的属性映射到我的 JSON 中的子属性?做你想做的事。

要使用它,首先[JsonConverter]向您的类添加一个属性,Viagem指定JsonPathConverter,如下所示:

[JsonConverter(typeof(JsonPathConverter))]
public partial class Viagem
{
    ...
}

然后,对于要从 JSON 反序列化的每个属性,添加一个[JsonProperty]属性,指定其在 JSON 中的点分隔路径。这里有一些例子:

    [JsonProperty("Viagem.Id")]
    public int Id { get; set; }

    [JsonProperty("Viagem.Status")]
    public string Status { get; set; }

    [JsonProperty("Viagem.Navio.Nome")]
    public string Navio_Nome { get; set; }

    [JsonProperty("Viagem.Navio.ImagemNavio")]
    public string ImagemNavio { get; set; }

    [JsonProperty("Viagem.Navio.Armador.Id")]
    public int? Armador_Id { get; set; }

    [JsonProperty("Viagem.Navio.Armador.Nome")]
    public string Armador_Nome { get; set; }

然后你可以像往常一样反序列化,它应该“正常工作”:

Viagem viagem = JsonConvert.DeserializeObject<Viagem>(json);

这是一个工作演示:https ://dotnetfiddle.net/repgUW

请注意,如果您需要将模型序列化回 JSON,则需要WriteJson在转换器中实现该方法。链接的问题线程中有另一个答案,它建议了一个实现,但我没有测试它。

于 2018-07-26T15:16:30.740 回答
0

不,您不想将 JSON 直接映射到数据库实体类中。您可以不遗余力地做一些属性和自定义序列化程序逻辑,但这是维护的噩梦(如果 JSON 或实体的结构发生变化怎么办?)。

而是生成一个类来反序列化这个 JSON,然后将该类的字段映射到您的实体。

于 2018-07-26T13:17:10.970 回答