views:

6068

answers:

4

I need to deserialize some JavaScript object represented in JSON to an appropriate C# class. Given the nice features of automatic properties, I would prefer having them in these classes as opposed to just having fields. Unfortunately, the .NET serialization engine (at least, by default) totally ignores automatic properties on deserialization and only cares about the backing field, which is obviously not present in the JavaScript object.

Given that there's no standard way to name backing fields and to be honest I don't even want to bother with the "let's create a JavaScript object that looks like it had C# backing fields" approach as it sounds a bit dirty, the only way I could serialize JavaScript fields to C# auto-properties if I could force the serialization engine to somehow ignore the backing field and use the property directly. Unfortunately, I can't figure out how this is done or if this can be done at all. Any ideas would be appreciated.

EDIT: Here's an example:

Javascript:

function Cat()
{
    this.Name = "Whiskers";
    this.Breed = "Tabby";
}
var cat = new Cat();

This is then serialized to "{Name: 'Whiskers'}".

The C# class:

[Serializable()]
public class Cat
{
    public string Name { get; set; }
    public string Breed { get; set; }
}

And the deserialization code, that fails:

new DataContractJsonSerializer(typeof(Cat)).ReadObject(inputStream);

And it is apparent from the exception that it fails because it is looking for the backing field.

EDIT2: Here's the exception, if that helps (no inner exceptions):

System.Runtime.Serialization.SerializationException

"The data contract type 'Test.Cat' cannot be deserialized because the required data members '<Name>k__BackingField, <Breed>k__BackingField' were not found."

+5  A: 

You can do this with JavaScriptSerializer found in the System.Web.Script.Serialization namespace:

JavaScriptSerializer serializer = new JavaScriptSerializer();
Cat c = serializer.Deserialize<Cat>(jsonString);

I have POCO objects with automatic properties and this works just fine.

EDIT: I wrote about JSON Serializers in .NET which compares this serializer with DataContractJsonSerializer.

aleemb
Yeah it looks like I'll be stuck with the good old JsonExSerializer http://code.google.com/p/jsonexserializer/
DrJokepu
What if the type I'm deserializing to is variable? Like if it comes as a string "Cat"?
towps
Oh you go over that in your article... cheers.
towps
A: 

I'm assuming you are passing data via a web service. If you are using the WebService class with the ScriptMethod attribute uncommented-out, the web service methods can read JSON natively. They even use the same JavaScriptSerializer that was mentioned above. If you are using WCF I'm a little more fuzzy on the logic.

But make sure your JSON object are returning data for EVERY property in your class. In your error, there is mention of a Breed property that is not in your example.

Also, on the JavaScript side, do to the dynamic nature of JavaScript it is easy to add new properties to your objects. This can sometimes lead to circular references. You should remove any extra data that you might have added (just as you are sending data via the web method, then add it again once you are done).

Chris Brandsma
No, unfortunately that's not a webservice. It is a simple ajax request that passes data in json to the server. Also, I'm pretty sure that there are no circular references. You're right that in the example there's no breed property but in my real code all properties are there. Thanks for the suggestions though :)
DrJokepu
On the JavaScript side of things: how are you converting the JavaScript object to JSON?
Chris Brandsma
@Chris Brandsma: The usual way, with json2.js: http://www.json.org/json2.js
DrJokepu
+8  A: 

You'd want to route the set operation to set_Name/set_Breed, rather than having the deserializer guess the backing field.

Solve this by adding some explicit mapping instead:

[DataContract]
public class Cat
{
    [DataMember]
    public string Name { get; set; }

    [DataMember]
    public string Breed { get; set; }
}
baretta
I'm trying out the JSON deserialization. I'm getting the following exception and as of yet cannot find much useful info on it to resolve. The data contract type 'MyType' cannot be deserialized because the required data members '<AllBlocks>k__BackingField, <Header>k__BackingField' were not found. Any ideas? Peace.
towps
ah my problem is inheritance. there are members in the base class that i have not provided values for in the JSON string...
towps
+1 First question I answered, but I did not have the rep to upvote this answer at the time!
amelvin
+1  A: 

baretta's answer solved the k__BackingField bloat for me. Just a tiny addendum that you can decorate this class to auto serialize into either XML or JSON in a similar way:

[Serializable, XmlRoot, DataContract]
public class Cat
{
  [XmlElement]
  [DataMember]
  public string Name { get; set; }
  [XmlElement]
  [DataMember]
  public string Breed { get; set; }
}

... and then use a DataContractJsonSerializer or XmlSerializer to prepare it for your endpoint.

amelvin