16

我尝试了一些变体,但没有运气在 GraphQL 中返回地图。所以我有以下两个对象:

public class Customer {

    private String name, age;
    // getters & setters
}

public class Person {

   private String type;
   private Map<String, Customer> customers;
   // getters & setters
}

我的架构如下所示:

type Customer {
   name: String!
   age:  String!
}

type Person {
  type: String!
  customers: [Customer!] // Here I tried all combination but had no luck, is there a Map type support for GQL?
}

有人可以告诉我如何实现这一点,以便 GraphQL 神奇地处理这个或替代方法。

非常感谢!

4

3 回答 3

12

正如您自己所指出的,GraphQL 中没有地图类型,主要是因为地图基本上是无类型数据(或具有动态结构的数据),因此不能很好地转换为 GraphQL 期望的静态类型。不过,您有几个选择。

1)您可以更改值类型以使其包含键,然后放弃地图并改用列表。这是您在自己的答案中采用的方法。我不会在这里详细介绍,因为您已经举例说明了。

2) 只要​​已知键和值 Java 类型(而不是 eg Object),您就可以将映射视为键值对列表。您可以创建一个类型来表示该对:

type Person {
  type: String!
  customers: [CustomerEntry!]
}

type CustomerEntry {
  key: String!
  value: Customer!
}

不利的一面是,您现在有更丑陋的查询:

{
   person {
     type
     customers {
       key
       value {
         name
       }
     }
   }
}

从好的方面来说,你保持类型安全和(主要)语义。可以继续嵌套这种方法来例如表示一个Map<String, Map<Long, Customer>>.

3)如果你有一个完全未知的类型,即Object,唯一的选择是将它视为一个复杂的标量。在 JavaScript 中,这种方法称为JSON 标量,因为它归结为填充任意 JSON 结构并将其视为标量。相同的方法可以在 Java 中实现。graphql-java 现在有一个扩展标量项目。这是他们的ObjectScalar(别名为JsonScalar)实现。

现在,如果您想表示一个类型,例如Map<String, Object>,您可以选择使用上面的键值对方法来表示它,只有值类型是 JSON 标量,或者您可以将整个映射表示为 JSON 标量。

事实上,您可以决定将任何地图(嗯,实际上是任何类型,但这没有用)表示为 JSON 标量。

type MapEntry {
  key: String!
  value: [ObjectScalar!]
}

scalar ObjectScalar

从好的方面来说,您现在可以准确地保持任何动态结构的形状。不利的一面是,由于它是一个标量,因此不可能进行子选择,并且您无法提前获取所有内容,而不知道里面有什么。

于 2018-01-27T04:32:11.450 回答
3

GraphQL 中没有地图类型(在 GitHub 上讨论)。

另一种方法是customers作为 a Listof Customers

public class Person {
   private String type;
   private List<Customer> customers;
}

并在Customer类中包含地图的键

public class Customer {
    private String key; // or another meaningful name
    private String name, age;
}

架构将基本保持不变。

type Customer {
   key: String! // or another meaningful name
   name: String!
   age: String!
}

type Person {
  type: String!
  customers: [Customer!]!
}
于 2017-12-07T17:01:56.493 回答
-1

以防万一 - 您始终可以将地图对象表示为 JSON 字符串(在我的情况下它很有帮助)。

public class Person {

    private String type;
    private Map<String, Customer> customers;
    // getters & setters
}

将会

type Person {
  type: String!
  customers: String!
}

之后不要忘记添加数据提取器以将其转换为 JSON。

public DataFetcher<String> fetchCustomers() {
        return environment -> {
            Person person = environment.getSource();
            try {
                ObjectMapper objectMapper = new ObjectMapper();
                return objectMapper.writeValueAsString(person.getCustomers());
            } catch (JsonProcessingException e) {
                log.error("There was a problem fetching the person!");
                throw new RuntimeException(e);
            }
        };
    }

它会返回:

"person": {
    "type": "2",
    "customers": "{\"VIP\":{\"name\":\"John\",\"age\":\"19\"},\"Platinum VIP\":{\"name\":\"Peter\",\"age\":\"65\"}}"
  }

之后,您可以像在客户端中使用典型的 JSON 字符串一样对客户进行操作。

于 2019-11-26T02:18:11.097 回答