Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Serialize List<KeyValuePair<string, string>> as JSON

I'm very new with JSON, please help!

I am trying to serialise a List<KeyValuePair<string, string>> as JSON

Currently:

[{"Key":"MyKey 1","Value":"MyValue 1"},{"Key":"MyKey 2","Value":"MyValue 2"}] 

Expected:

[{"MyKey 1":"MyValue 1"},{"MyKey 2":"MyValue 2"}] 

I referred to some examples from this and this.

This is my KeyValuePairJsonConverter : JsonConverter

public class KeyValuePairJsonConverter : JsonConverter {     public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)     {         List<KeyValuePair<object, object>> list = value as List<KeyValuePair<object, object>>;         writer.WriteStartArray();         foreach (var item in list)         {             writer.WriteStartObject();             writer.WritePropertyName(item.Key.ToString());             writer.WriteValue(item.Value.ToString());             writer.WriteEndObject();         }         writer.WriteEndArray();     }      public override bool CanConvert(Type objectType)     {         return objectType == typeof(List<KeyValuePair<object, object>>);     }      public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)     {         var jsonObject = JObject.Load(reader);         var target = Create(objectType, jsonObject);         serializer.Populate(jsonObject.CreateReader(), target);         return target;     }      private object Create(Type objectType, JObject jsonObject)     {         if (FieldExists("Key", jsonObject))         {             return jsonObject["Key"].ToString();         }          if (FieldExists("Value", jsonObject))         {             return jsonObject["Value"].ToString();         }         return null;     }      private bool FieldExists(string fieldName, JObject jsonObject)     {         return jsonObject[fieldName] != null;     } } 

I am calling it from a WebService method like this

List<KeyValuePair<string, string>> valuesList = new List<KeyValuePair<string, string>>(); Dictionary<string, string> valuesDict = SomeDictionaryMethod();  foreach(KeyValuePair<string, string> keyValue in valuesDict) {     valuesList.Add(keyValue); }  JsonSerializerSettings jsonSettings = new JsonSerializerSettings { Converters = new [] {new KeyValuePairJsonConverter()} }; string valuesJson = JsonConvert.SerializeObject(valuesList, jsonSettings); 
like image 771
maryhadalittlelamb Avatar asked Jan 06 '17 09:01

maryhadalittlelamb


2 Answers

You can use Newtonsoft and dictionary:

    var dict = new Dictionary<int, string>();     dict.Add(1, "one");     dict.Add(2, "two");      var output = Newtonsoft.Json.JsonConvert.SerializeObject(dict); 

The output is :

{"1":"one","2":"two"} 

Edit

Thanks to @Sergey Berezovskiy for the information.

You are currently using Newtonsoft, so just change your List<KeyValuePair<object, object>> to Dictionary<object,object> and use the serialize and deserialize method from the package.

like image 82
OrcusZ Avatar answered Sep 20 '22 03:09

OrcusZ


So I didn't want to use anything but native c# to solve a similar issue and for reference this was using .net 4, jquery 3.2.1 and backbone 1.2.0.

My issues was that the List<KeyValuePair<...>> would process out of the controller into a backbone model but when I saved that model the controller could not bind List.

public class SomeModel {     List<KeyValuePair<int, String>> SomeList { get; set; } }  [HttpGet] SomeControllerMethod() {     SomeModel someModel = new SomeModel();     someModel.SomeList = GetListSortedAlphabetically();     return this.Json(someModel, JsonBehavior.AllowGet); } 

network capture:

"SomeList":[{"Key":13,"Value":"aaab"},{"Key":248,"Value":"aaac"}] 

But even though this set SomeList properly in the backing model.js trying to save the model without any changes to it would cause the binding SomeModel object to have the same length as the parameters in the request body but all the keys and values were null:

[HttpPut] SomeControllerMethod([FromBody] SomeModel){     SomeModel.SomeList; // Count = 2, all keys and values null. } 

The only things I could find is that KeyValuePair is a structure and not something that can be instantiated in this manner. What I ended up doing is the following:

  • Add a Model wrapper somewhere that contains key, value fields:

    public class KeyValuePairWrapper {     public int Key { get; set; }     public String Value { get; set; }      //default constructor will be required for binding, the Web.MVC binder will invoke this and set the Key and Value accordingly.     public KeyValuePairWrapper() { }      //a convenience method which allows you to set the values while sorting     public KeyValuePairWrapper(int key, String value)     {         Key = key;         Value = value;     } } 
  • Set up your binding class model to accept your custom wrapper object.

    public class SomeModel {     public List<KeyValuePairWrapper> KeyValuePairList{ get; set };  } 
  • Get some json data out of a controller

    [HttpGet] SomeControllerMethod() {     SomeModel someModel = new SomeModel();     someModel.KeyValuePairList = GetListSortedAlphabetically();     return this.Json(someModel, JsonBehavior.AllowGet); } 
  • Do something at a later time, maybe model.save(null, ...) is invoked

    [HttpPut] SomeControllerMethod([FromBody] SomeModel){     SomeModel.KeyValuePairList ; // Count = 2, all keys and values are correct. } 
like image 21
steven87vt Avatar answered Sep 22 '22 03:09

steven87vt