C sharp and JSON
General notes
Section titled General notes- At least as of mid 2024, the built-in
System.Text.Json
stuff seems pretty powerful. - The official Microsoft documentation is good (although sometimes long).
- Note to self: I worked somewhat extensively with JSON stuff for Skeleseller, so if I’m ever looking for code in the future, check there. To anyone else reading this, the source is closed at the time of writing. 😢
Serializing read-only properties that are collections
Section titled Serializing read-only properties that are collectionsI had code like this:
The way this is written, if serialized/deserialized, the Dictionary
’s values will always be what’s hard-coded there. To fix it, there are two options:
- Add a
set;
to the property. This will result in linting rule CA2227 telling you not to do that. - Preferred: add
[JsonObjectCreationHandling(JsonObjectCreationHandling.Populate)]
just above theDictionary
(reference)
Troubleshooting
Section titled TroubleshootingDeserialization problems
Section titled Deserialization problemsThis error is sort of hard to decipher at first. It means that you’re trying to load data from a JSON file that doesn’t correspond to a constructor or field. For example, suppose I have this in JSON:
…and this constructor in C#:
There are two problems above:
- The JSON has the name
RewardPoints
but C# uses the namerewardSize
. You can fix this simply by renamingrewardSize
→rewardPoints
(since, as the error mentions, the match is case-insensitive). - The JSON has a
Progress
dictionary but theQuest
constructor doesn’t take that in. For this fix, it made more sense to make aProgress
property (public Dictionary<ItemId, int> Progress { get; init; }
).
In conclusion, C# just needs to know how to deserialize what’s found in JSON either via a constructor/property/field.
A custom converter is serializing incorrectly
Section titled A custom converter is serializing incorrectlyThis is the fault of the custom converter I wrote:
Depending on what you want, there are two different fixes:
- If the JSON you want is
"Item": { "id": 1 }
, then surround the call towriter.WriteNumber
withwriter.WriteStartObject();
andwriter.WriteEndObject();
- This also requires a change to the reader (see below); you generally need to loop on
reader.Read()
and check for whenreader.TokenType
isJsonTokenType.StartObject
,JsonTokenType.EndObject
, orJsonTokenType.PropertyName
.
- This also requires a change to the reader (see below); you generally need to loop on
- If the JSON you want is
"Item": 1
, then replacewriter.WriteNumber
withwriter.WriteNumberValue
.
Similarly, if you wanted to write arrays, you need to explicitly call WriteStartArray
and WriteEndArray
.
Example reader change
Section titled Example reader changeThis is kind of ridiculous, but here’s how you would read the { "id": 1 }
object. I didn’t need a loop since I only write one value: