json.netカスタマイズされたマルチステートオブジェクトシリアル化、追加のタイプフィールド
json.netの
設定/誘導体の設定/逆のシーケンシャル化は非常に便利です。ただし、この方法では、シリアル化されたJSONに追加のTypeNameHandling.Auto
フィールドが導入され、オーバーヘッドが増加します。 $type
コーディングタイプの属性を備えたカスタムベースクラス:
コンテナクラスに保存されたサブタイプ情報の代わりに、基本クラスへの属性として追加する方が良いです。
この属性は、オブジェクトのオブジェクトのタイプを反映し、列挙値にエンコードします。辞書は、型を列挙された値にマッピングするためにそれ自体で実装する必要があります。
<code class="language-csharp">[JsonConverter(typeof(SubTypeClassConverter))] public class SubTypeClassBase { [JsonConverter(typeof(StringEnumConverter))] public SubType Type { get { return typeToSubType[GetType()]; } } }</code>
typeToSubType
:
メソッドは単独で実装する必要があり、列挙値に従って対応するタイプを返します。
JsonConverter
利点:
<code class="language-csharp">public class SubTypeClassConverter : JsonConverter { // 读取 JSON 并确定实际类型 public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) { var token = JToken.Load(reader); var typeToken = token["Type"]; var actualType = SubTypeClassBase.GetTypeFromSubType(typeToken.ToObject<SubType>(serializer)); // 使用 GetTypeFromSubType 方法 if (existingValue == null || existingValue.GetType() != actualType) { // 创建实际的对象实例 var contract = serializer.ContractResolver.ResolveContract(actualType); existingValue = contract.DefaultCreator(); } // 使用 "Populate" 方法避免无限递归 using (var subReader = token.CreateReader()) { serializer.Populate(subReader, existingValue); } return existingValue; } // ...其他方法 (WriteJson 等) ... }</code>
コンテナとは無関係に、クラス内の属性にエンコードするGetTypeFromSubType
サブタイプの情報。
循環化は動的に処理され、クラス構造がシリアル化を損なうことなく将来クラス構造を変更できるようにします。 JSON出力の冗長
フィールドを排除し、そのサイズを縮小します。$type
列挙には、可能なすべてのサブタイプを含める必要があります。 メソッドは逆マッピングを実行する必要があります。
以上が「$type」フィールドを使用せずに、Json.Net でポリモーフィックな子オブジェクトのカスタム シリアル化を実現するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。