gpt4 book ai didi

c# - F# 与 C# 中的反序列化

转载 作者:行者123 更新时间:2023-11-30 15:50:56 25 4
gpt4 key购买 nike

我有以下 json:

{
"data": [
{
"timestamp": "2019-11-07T00:23:52.095Z",
"symbol": "XBTUSD",
"side": "Buy",
"size": 1,
"price": 9356.5,
"tickDirection": "PlusTick",
"trdMatchID": "01476235-ad89-1777-9067-8ce6d0e29992",
"grossValue": 10688,
"homeNotional": 0.00010688,
"foreignNotional": 1
}
]
}

最后 3 个字段是可选的。

当我在 C# 中反序列化它时,我执行以下操作:

public class Trade
{
public DateTime Timestamp;
public string Symbol;
public string Side;
public long Size;
public long Price;
public long? GrossValue;
public float? HomeNotional;
public float? ForeignNotional;
}

public class TradeContainer
{
public Trade[] Data;
}

var j = JsonConvert.DeserializeObject<TradeContainer>(x);

一切都很好。

在 F# 中,我这样做:

type Trade =
{
Timestamp : DateTime
Symbol : string
Side : string
Size : int64
Price : int64
GrossValue : int64 option
HomeNotional : float option
ForeignNotional : float option
}

type TradeContainer =
{
Data : Trade[]
}

let t = JsonConvert.DeserializeObject<TradeContainer>(x)

但它会失败。但是,如果我删除 Trade 类型中的 option 关键字,那么它会正确反序列化。

我得到的错误是:

Newtonsoft.Json.JsonSerializationException: Unexpected property 'homeNotional' found when reading union. Path 'data[0].homeNotional', line 1, position 233. at at Newtonsoft.Json.Converters.DiscriminatedUnionConverter.ReadJson(JsonReader reader, Type objectType, Object existingValue, JsonSerializer serializer)

为什么不同?但我又该如何实现这一目标,因为我需要计划某些值可能不存在?

最佳答案

问题出在 JSON.NET 的 DU 序列化实现上。 It's not idiomatic并基本上转储案例和字段:

type Shape =
| Rectangle of width : float * length : float
| Circle of radius : float
| Empty

[<EntryPoint>]
let main argv =

let shape1 = Rectangle(1.3, 10.0)

let json = JsonConvert.SerializeObject(shape1)
// {
// "Case": "Rectangle",
// "Fields": [
// 1.3,
// 10.0
// ]
// }

反序列化器需要相同的结构。

Isaac Abraham 创建了 an idiomatic custom converter应该改用它:

let settings = new JsonSerializerSettings()
settings.Converters.Add(IdiomaticDuConverter())

let t = JsonConvert.DeserializeObject<TradeContainer>(json,settings)

IdiomaticDuConverter 的代码是:

namespace Newtonsoft.Json.Converters

open Microsoft.FSharp.Reflection
open Newtonsoft.Json
open System

type IdiomaticDuConverter() =
inherit JsonConverter()

[<Literal>]
let discriminator = "__Case"
let primitives = Set [ JsonToken.Boolean; JsonToken.Date; JsonToken.Float; JsonToken.Integer; JsonToken.Null; JsonToken.String ]

let writeValue (value:obj) (serializer:JsonSerializer, writer : JsonWriter) =
if value.GetType().IsPrimitive then writer.WriteValue value
else serializer.Serialize(writer, value)

let writeProperties (fields : obj array) (serializer:JsonSerializer, writer : JsonWriter) =
fields |> Array.iteri (fun index value ->
writer.WritePropertyName(sprintf "Item%d" index)
(serializer, writer) |> writeValue value)

let writeDiscriminator (name : string) (writer : JsonWriter) =
writer.WritePropertyName discriminator
writer.WriteValue name

override __.WriteJson(writer, value, serializer) =
let unionCases = FSharpType.GetUnionCases(value.GetType())
let unionType = value.GetType()
let case, fields = FSharpValue.GetUnionFields(value, unionType)
let allCasesHaveValues = unionCases |> Seq.forall (fun c -> c.GetFields() |> Seq.length > 0)

match unionCases.Length, fields, allCasesHaveValues with
| 2, [||], false -> writer.WriteNull()
| 1, [| singleValue |], _
| 2, [| singleValue |], false -> (serializer, writer) |> writeValue singleValue
| 1, fields, _
| 2, fields, false ->
writer.WriteStartObject()
(serializer, writer) |> writeProperties fields
writer.WriteEndObject()
| _ ->
writer.WriteStartObject()
writer |> writeDiscriminator case.Name
(serializer, writer) |> writeProperties fields
writer.WriteEndObject()

override __.ReadJson(reader, destinationType, _, _) =
let parts =
if reader.TokenType <> JsonToken.StartObject then [| (JsonToken.Undefined, obj()), (reader.TokenType, reader.Value) |]
else
seq {
yield! reader |> Seq.unfold (fun reader ->
if reader.Read() then Some((reader.TokenType, reader.Value), reader)
else None)
}
|> Seq.takeWhile(fun (token, _) -> token <> JsonToken.EndObject)
|> Seq.pairwise
|> Seq.mapi (fun id value -> id, value)
|> Seq.filter (fun (id, _) -> id % 2 = 0)
|> Seq.map snd
|> Seq.toArray

let values =
parts
|> Seq.filter (fun ((_, keyValue), _) -> keyValue <> (discriminator :> obj))
|> Seq.map snd
|> Seq.filter (fun (valueToken, _) -> primitives.Contains valueToken)
|> Seq.map snd
|> Seq.toArray

let case =
let unionCases = FSharpType.GetUnionCases(destinationType)
let unionCase =
parts
|> Seq.tryFind (fun ((_,keyValue), _) -> keyValue = (discriminator :> obj))
|> Option.map (snd >> snd)
match unionCase with
| Some case -> unionCases |> Array.find (fun f -> f.Name :> obj = case)
| None ->
// implied union case
match values with
| [| null |] -> unionCases |> Array.find(fun c -> c.GetFields().Length = 0)
| _ -> unionCases |> Array.find(fun c -> c.GetFields().Length > 0)

let values =
case.GetFields()
|> Seq.zip values
|> Seq.map (fun (value, propertyInfo) -> Convert.ChangeType(value, propertyInfo.PropertyType))
|> Seq.toArray

FSharpValue.MakeUnion(case, values)

override __.CanConvert(objectType) =
FSharpType.IsUnion objectType &&
not (objectType.IsGenericType &&
typedefof<list<_>> = objectType.GetGenericTypeDefinition())

关于c# - F# 与 C# 中的反序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58748946/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com