F#Data JSON类型提供程序:如何处理可以是数组或属性的JSON属性?

时间:2016-05-01 07:23:54

标签: json f# type-providers f#-data fsharp.data.typeproviders

我正在使用F#Data库中的JSON类型提供程序从API访问JSON文档。文档包含一个属性(让我们称之为'car'),有时候是一个对象数组,有时候是一个对象。它是这样的:

'car': [
  { ... },
  { ... }
]

或者这个:

'car': { ... }

{ ... }中的对象在两种情况下具有相同的结构。

JSON类型提供程序指示属性的类型为:

JsonProvider<"../data/sample.json">.ArrayOrCar

其中sample.json是我的示例文档。

我的问题是:我如何判断该属性是否是一个数组(以便我可以将其作为数组处理)或单个对象(以便我可以将其作为对象处理)?

更新 简化的示例JSON将如下所示:

{
  "set": [
    {
      "car": [
        {
          "brand": "BMW" 
        },
        {
          "brand": "Audi"
        }
      ]
    },
    {
      "car": {
        "brand": "Toyota"
      }
    } 
  ]
}

使用以下代码,我们会指出doc.Set.[0].Car的类型为JsonProvider<...>.ArrayOrCar

type example = JsonProvider<"sample.json">
let doc = example.GetSample()
doc.Set.[0].Car

2 个答案:

答案 0 :(得分:5)

如果数组中JSON值的类型与直接嵌套的JSON值的类型相同,那么JSON类型提供程序实际上将统一这两种类型,因此您可以使用相同的函数处理它们。

使用最小的JSON文档作为示例,以下工作:

type J = JsonProvider<"sample.json">

// The type `J.Car` is the type of the car elements in the array    
// but also of the car record directly nested in the "car" field
let printBrand (car:J.Car) =
  printfn "%s" car.Brand

// Now you can use pattern matching to check if the current element
// has an array of cars or just a single record - then you can call
// `printBrand` either on all cars or on just a single car
let doc = J.GetSample()
for set in doc.Set do
  match set.Car.Record, set.Car.Array with
  | Some c, _ -> printBrand c
  | _, Some a -> for c in a do printBrand c
  | _ -> failwith "Wrong input"

答案 1 :(得分:1)

在玩完图书馆之后,我发现你可以做这样的事情:

let car = doc.Set.[0].Car
let processObject (car:example.ArrayOrCar) = 
    match car.Array with
    | Some a -> printfn "do stuff with an array %A" a
    | None ->  match car.Record with 
               | Some c -> printfn "do stuff with an object %A" c
               | None -> printfn "fail here?"

要处理整个Set[],您可以使用Array.map之类的内容将processObject应用于每个元素。