c# Ändern Sie den Namen von "Key" und "Value", wenn Sie JSON.NET zum Serialisieren eines komplexen Dictionary verwenden




serialization (2)

Ich stoße auf ein seltsames Problem mit der Serialisierung von Daten mit json.net. Hauptsächlich versuche ich, die 'Schlüssel'- und' Wert'-Namen in dem ausgehenden JSON als etwas Beschreibenderes umzubenennen. Insbesondere möchte ich, dass der IRequest-bezogene 'Schlüssel' 'Request' genannt wird und der IQuoteTimeSeries 'Value' 'DataSeries' ist.

Hinweis: Dies wird nicht deserialisiert. Es wird nur bei der Datenanalyse auf der Webseite verwendet.

Das Datenrepository-Objekt, das ich serialisiere, ist ein Dictionary<IRequest, IQuoteTimeSeries> -Objekt. Die IRequest stellt eine bestimmte Anforderung für Daten und die IQuoteTimeSeries ist das Objekt, das die zurückgegebenen Daten als SortedDictionary<DateTime, IQuote> . Dies ist eine Reihe von Daten, die nach Zeitmarken sortiert sind. In diesem Beispiel habe ich der Kürze halber nur ein Element in der TimeSeries, aber in den meisten Fällen würde es viele Elemente geben.

Alles muss zusammen organisiert, serialisiert und verschickt werden, um von JavaScript konsumiert zu werden.

Hier ist der grundlegende Code für diese Objekte;

[JsonArray]
public class QuoteRepository : Dictionary<IRequest, IQuoteTimeSeries>, IQuoteRepository
{
    public QuoteRepository() { }   

    public void AddRequest(IRequest request)
    {
        if (!this.ContainsKey(request))
        {
            IQuoteTimeSeries tSeries = new QuoteTimeSeries(request);
            this.Add(request, tSeries);
        }
    }

    public void AddQuote(IRequest request, IQuote quote)
    {        
        if (!this.ContainsKey(request))
        {
            QuoteTimeSeries tSeries = new QuoteTimeSeries(request);
            this.Add(request, tSeries);
        }        
        this[request].AddQuote(quote);        
    }    

    IEnumerator<IQuoteTimeSeries>  Enumerable<IQuoteTimeSeries>.GetEnumerator()
    {
        return this.Values.GetEnumerator();
    }
}

Eine Zitat-Zeitreihe sieht so aus;

[JsonArray]
public class QuoteTimeSeries: SortedDictionary<DateTime, IQuote>, IQuoteTimeSeries
{  
    public QuoteTimeSeries(IRequest request)
    {
        Request = request;
    }
    public IRequest Request { get; }
    public void AddQuote(IQuote quote)
    {
        this[quote.QuoteTimeStamp] = quote;
    }

    public void MergeQuotes(IQuoteTimeSeries quotes)
    {
        foreach (IQuote item in quotes)
        {
            this[item.QuoteTimeStamp] = item;
        }
    }

    IEnumerator<IQuote> IEnumerable<IQuote>.GetEnumerator()
    {
        return this.Values.GetEnumerator();
    }

}

Der zur Serialisierung verwendete Code ist ziemlich einfach:

IQuoteRepository quotes = await requests.GetDataAsync();

JsonSerializerSettings settings = new JsonSerializerSettings
{
    ContractResolver = QuoteRepositoryContractResolver.Instance,
    NullValueHandling = NullValueHandling.Ignore
};

return Json<IQuoteRepository>(quotes, settings);

Ich fügte einen Vertragsresolver mit der Absicht hinzu, das Schreiben des Eigentums außer Kraft zu setzen. Der property.PropertyName = Code wird getroffen und die Eigenschaftsnamen werden geändert, aber der Ausgabe-JSON ist davon nicht betroffen.

public class QuoteRepositoryContractResolver : DefaultContractResolver
{
    public static readonly QuoteRepositoryContractResolver Instance = new QuoteRepositoryContractResolver();

    protected override JsonProperty CreateProperty(MemberInfo member, MemberSerialization memberSerialization)
    {
        JsonProperty property = base.CreateProperty(member, memberSerialization);

        if (property.DeclaringType == typeof(KeyValuePair<IRequest, IQuoteTimeSeries>))
        {
            if (property.PropertyName.Equals("Key", StringComparison.OrdinalIgnoreCase))
            {
                property.PropertyName = "Request";
            }
            else if (property.PropertyName.Equals("Value", StringComparison.OrdinalIgnoreCase))
            {
                property.PropertyName = "Data";
            }
        }
        else if (property.DeclaringType == typeof(KeyValuePair<DateTime, IQuote>))
        {
            if (property.PropertyName.Equals("Key", StringComparison.OrdinalIgnoreCase))
            {
                property.PropertyName = "TimeStamp";
            }
            else if (property.PropertyName.Equals("Value", StringComparison.OrdinalIgnoreCase))
            {
                property.PropertyName = "Quote";
            }
        }
        return property;
    }
}

Die Ausgabe JSON ist ungerade. Die Elemente "Schlüssel" und "Wert" sind vollständig unverändert, obwohl ich ihre Namen im Code geändert habe.

[
    {
        "Key": {
            "QuoteDate": "2016-05-12T00:00:00-04:00",
            "QuoteType": "Index",
            "Symbol": "SPY",
            "UseCache": true
        },
        "Value": [
            {
                "Key": "2016-05-11T16:00:01-04:00",
                "Value": {
                    "Description": "SPDR S&amp;P 500",
                    "High": 208.54,
                    "Low": 206.50,
                    "Change": -1.95,
                    "ChangePer": -0.94,
                    "Price": 206.50,
                    "QuoteTimeStamp": "2016-05-11T16:00:01-04:00",
                    "Symbol": "SPY"
                }
            }
        ]
    },
    {
        "Key": {
            "QuoteDate": "2016-05-12T00:00:00-04:00",
            "QuoteType": "Stock",
            "Symbol": "GOOG",
            "UseCache": true
        },
        "Value": [
            {
                "Key": "2016-05-11T16:00:00-04:00",
                "Value": {
                    "Description": "Alphabet Inc.",
                    "High": 724.48,
                    "Low": 712.80,
                    "Change": -7.89,
                    "ChangePer": -1.09,
                    "Price": 715.29,
                    "QuoteTimeStamp": "2016-05-11T16:00:00-04:00",
                    "Symbol": "GOOG"
                }
            }
        ]
    }
]

Weiß jemand, wie man die Elemente "Schlüssel" und "Wert" richtig ändert?


Das Konvertieren Ihres Wörterbuchs von Dictionary<IRequest, IQuoteTimeSeries> zu List<KeyValuePair<IRequest, IQuoteTimeSeries>> sollte ebenfalls funktionieren.


Eine Möglichkeit, dies zu lösen, besteht darin, einen benutzerdefinierten JsonConverter für Ihre wörterbuchbasierten Klassen zu verwenden. Der Code ist eigentlich ziemlich einfach.

public class CustomDictionaryConverter<K, V> : JsonConverter
{
    private string KeyPropertyName { get; set; }
    private string ValuePropertyName { get; set; }

    public CustomDictionaryConverter(string keyPropertyName, string valuePropertyName)
    {
        KeyPropertyName = keyPropertyName;
        ValuePropertyName = valuePropertyName;
    }

    public override bool CanConvert(Type objectType)
    {
        return typeof(IDictionary<K, V>).IsAssignableFrom(objectType);
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        IDictionary<K, V> dict = (IDictionary<K, V>)value;
        JArray array = new JArray();
        foreach (var kvp in dict)
        {
            JObject obj = new JObject();
            obj.Add(KeyPropertyName, JToken.FromObject(kvp.Key, serializer));
            obj.Add(ValuePropertyName, JToken.FromObject(kvp.Value, serializer));
            array.Add(obj);
        }
        array.WriteTo(writer);
    }

    public override bool CanRead
    {
        get { return false; }
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}

Wenn es Zeit ist zu serialisieren, fügen Sie die Konverter zu den JsonSerializerSettings wie JsonSerializerSettings :

JsonSerializerSettings settings = new JsonSerializerSettings
{
    Converters = new List<JsonConverter>
    {
        new CustomDictionaryConverter<IRequest, IQuoteTimeSeries>("Request", "Data"),
        new CustomDictionaryConverter<DateTime, IQuote>("TimeStamp", "Quote")
    },
    Formatting = Formatting.Indented
};

string json = JsonConvert.SerializeObject(repo, settings);

Geige: https://dotnetfiddle.net/roHEtx





json.net