JSON.NET でカスタム JsonConverter を実装するにはどうすればいいですか? 質問する

JSON.NET でカスタム JsonConverter を実装するにはどうすればいいですか? 質問する

ここで示されているJSON.netの例を拡張しようとしていますhttp://james.newtonking.com/projects/json/help/CustomCreationConverter.html

基本クラス/インターフェースから派生した別のサブクラスがあります

public class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
}

public class Employee : Person
{
    public string Department { get; set; }
    public string JobTitle { get; set; }
}

public class Artist : Person
{
    public string Skill { get; set; }
}

List<Person> people  = new List<Person>
{
    new Employee(),
    new Employee(),
    new Artist(),
};

次の Json をリスト < Person > に逆シリアル化するにはどうすればよいですか?

[
  {
    "Department": "Department1",
    "JobTitle": "JobTitle1",
    "FirstName": "FirstName1",
    "LastName": "LastName1"
  },
  {
    "Department": "Department2",
    "JobTitle": "JobTitle2",
    "FirstName": "FirstName2",
    "LastName": "LastName2"
  },
  {
    "Skill": "Painter",
    "FirstName": "FirstName3",
    "LastName": "LastName3"
  }
]

TypeNameHandling JsonSerializerSettings は使いたくありません。特に、これを処理するためのカスタム JsonConverter 実装を探しています。これに関するドキュメントや例は、ネット上でほとんど見つかりません。JsonConverter でオーバーライドされた ReadJson() メソッドの実装を正しく実行できないようです。

ベストアンサー1

標準の を使用すると、正しい型 (または)CustomCreationConverterを生成する方法を見つけるのに苦労しました。これを判断するには JSON を分析する必要があり、 メソッドを使用してこれを行うための組み込みの方法がないためです。PersonEmployeeCreate

型変換に関するディスカッション スレッドを見つけたのですが、そこに答えが載っていました。こちらがリンクです:型変換(アーカイブリンク)

必要なのは、 をサブクラス化しJsonConverter、 メソッドをオーバーライドして、を受け入れるReadJson新しい抽象メソッドを作成することです。CreateJObject

JObject クラスは、JSON オブジェクトを読み込む手段を提供し、このオブジェクト内のデータへのアクセスを提供します。

オーバーライドされたReadJsonメソッドは を作成し、インスタンスを渡してメソッド (派生コンバーター クラスによって実装)JObjectを呼び出します。CreateJObject

このJObjectインスタンスを分析して、特定のフィールドの存在をチェックすることで正しいタイプを判別できます。

string json = "[{
        \"Department\": \"Department1\",
        \"JobTitle\": \"JobTitle1\",
        \"FirstName\": \"FirstName1\",
        \"LastName\": \"LastName1\"
    },{
        \"Department\": \"Department2\",
        \"JobTitle\": \"JobTitle2\",
        \"FirstName\": \"FirstName2\",
        \"LastName\": \"LastName2\"
    },
        {\"Skill\": \"Painter\",
        \"FirstName\": \"FirstName3\",
        \"LastName\": \"LastName3\"
    }]";

List<Person> persons = 
    JsonConvert.DeserializeObject<List<Person>>(json, new PersonConverter());

...

public class PersonConverter : JsonCreationConverter<Person>
{
    protected override Person Create(Type objectType, JObject jObject)
    {
        if (FieldExists("Skill", jObject))
        {
            return new Artist();
        }
        else if (FieldExists("Department", jObject))
        {
            return new Employee();
        }
        else
        {
            return new Person();
        }
    }

    private bool FieldExists(string fieldName, JObject jObject)
    {
        return jObject[fieldName] != null;
    }
}

public abstract class JsonCreationConverter<T> : JsonConverter
{
    /// <summary>
    /// Create an instance of objectType, based properties in the JSON object
    /// </summary>
    /// <param name="objectType">type of object expected</param>
    /// <param name="jObject">
    /// contents of JSON object that will be deserialized
    /// </param>
    /// <returns></returns>
    protected abstract T Create(Type objectType, JObject jObject);

    public override bool CanConvert(Type objectType)
    {
        return typeof(T).IsAssignableFrom(objectType);
    }

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

    public override object ReadJson(JsonReader reader, 
                                    Type objectType, 
                                     object existingValue, 
                                     JsonSerializer serializer)
    {
        // Load JObject from stream
        JObject jObject = JObject.Load(reader);

        // Create target object based on JObject
        T target = Create(objectType, jObject);

        // Populate the object properties
        serializer.Populate(jObject.CreateReader(), target);

        return target;
    }
}

おすすめ記事