Java 使用 Graphql 搭建查詢服務(wù)詳解
背景
隨著React的開源,facebook相繼開源了很多相關(guān)的項目,這些項目在他們內(nèi)部已經(jīng)使用了多年,其中引起我注意的就是本次討論的是graphql,目前官方只有nodejs版,由于很多公司的后臺技術(shù)棧都是Java,所以便有了graphql的java版實現(xiàn),在github上可以找到,廢話不多說,直接看代碼吧,具體介紹還是去看官網(wǎng)吧,不然就跑題了。
GraphQLSchema
Schema相當于一個數(shù)據(jù)庫,它有很多GraphQLFieldDefinition組成,F(xiàn)ield相當于數(shù)據(jù)庫表/視圖,每個表/視圖又由名稱、查詢參數(shù)、數(shù)據(jù)結(jié)構(gòu)、數(shù)據(jù)組成.
1) 先定義一個數(shù)據(jù)結(jié)構(gòu)(GraphQLOutputType)字段,然后定義一個初始化方法
private GraphQLOutputType userType;
private void initOutputType() {
/**
* 會員對象結(jié)構(gòu)
*/
userType = newObject()
.name("User")
.field(newFieldDefinition().name("id").type(GraphQLInt).build())
.field(newFieldDefinition().name("age").type(GraphQLInt).build())
.field(newFieldDefinition().name("sex").type(GraphQLInt).build())
.field(newFieldDefinition().name("name").type(GraphQLString).build())
.field(newFieldDefinition().name("pic").type(GraphQLString).build())
.build();
}
2)再定義兩個表/視圖,它包括名稱,查詢參數(shù),數(shù)據(jù)結(jié)構(gòu),以及數(shù)據(jù)檢索器
/**
* 查詢單個用戶信息
* @return
*/
private GraphQLFieldDefinition createUserField() {
return GraphQLFieldDefinition.newFieldDefinition()
.name("user")
.argument(newArgument().name("id").type(GraphQLInt).build())
.type(userType)
.dataFetcher(environment -> {
// 獲取查詢參數(shù)
int id = environment.getArgument("id");
// 執(zhí)行查詢, 這里隨便用一些測試數(shù)據(jù)來說明問題
User user = new User();
user.setId(id);
user.setAge(id + 15);
user.setSex(id % 2);
user.setName("Name_" + id);
user.setPic("pic_" + id + ".jpg");
return user;
})
.build();
}
/**
* 查詢多個會員信息
* @return
*/
private GraphQLFieldDefinition createUsersField() {
return GraphQLFieldDefinition.newFieldDefinition()
.name("users")
.argument(newArgument().name("page").type(GraphQLInt).build())
.argument(newArgument().name("size").type(GraphQLInt).build())
.argument(newArgument().name("name").type(GraphQLString).build())
.type(new GraphQLList(userType))
.dataFetcher(environment -> {
// 獲取查詢參數(shù)
int page = environment.getArgument("page");
int size = environment.getArgument("size");
String name = environment.getArgument("name");
// 執(zhí)行查詢, 這里隨便用一些測試數(shù)據(jù)來說明問題
List<User> list = new ArrayList<>(size);
for (int i = 0; i < size; i++) {
User user = new User();
user.setId(i);
user.setAge(i + 15);
user.setSex(i % 2);
user.setName(name + "_" + page + "_" + i);
user.setPic("pic_" + i + ".jpg");
list.add(user);
}
return list;
})
.build();
}
3)接著定義一個Schema,并將其初始化,它包含一個名稱,以及一個或多個表/視圖(Field)
private GraphQLSchema schema;
public GraphSchema() {
initOutputType();
schema = GraphQLSchema.newSchema().query(newObject()
.name("GraphQuery")
.field(createUsersField())
.field(createUserField())
.build()).build();
}
4)完成以上步驟之后,還需要定義一個model,類名不限,但是結(jié)構(gòu)需要滿足前面定義的數(shù)據(jù)結(jié)構(gòu),而且必須是public的
public class User {
private int id;
private int age;
private int sex;
private String name;
private String pic;
// getter, setter...
}
5)之后寫一個main方法,來測試一下
public static void main(String[] args) {
GraphQLSchema schema = new GraphSchema().getSchema();
String query1 = "{users(page:2,size:5,name:\"john\") {id,sex,name,pic}}";
String query2 = "{user(id:6) {id,sex,name,pic}}";
String query3 = "{user(id:6) {id,sex,name,pic},users(page:2,size:5,name:\"john\") {id,sex,name,pic}}";
Map<String, Object> result1 = (Map<String, Object>) new GraphQL(schema).execute(query1).getData();
Map<String, Object> result2 = (Map<String, Object>) new GraphQL(schema).execute(query2).getData();
Map<String, Object> result3 = (Map<String, Object>) new GraphQL(schema).execute(query3).getData();
// 查詢用戶列表
System.out.println(result1);
// 查詢單個用戶
System.out.println(result2);
// 單個用戶、跟用戶列表一起查
System.out.println(result3);
}
輸出:
{users=[{id=0, sex=0, name=john_2_0, pic=pic_0.jpg}, {id=1, sex=1, name=john_2_1, pic=pic_1.jpg}, {id=2, sex=0, name=john_2_2, pic=pic_2.jpg}, {id=3, sex=1, name=john_2_3, pic=pic_3.jpg}, {id=4, sex=0, name=john_2_4, pic=pic_4.jpg}]}
{user={id=6, sex=0, name=Name_6, pic=pic_6.jpg}}
{user={id=6, sex=0, name=Name_6, pic=pic_6.jpg}, users=[{id=0, sex=0, name=john_2_0, pic=pic_0.jpg}, {id=1, sex=1, name=john_2_1, pic=pic_1.jpg}, {id=2, sex=0, name=john_2_2, pic=pic_2.jpg}, {id=3, sex=1, name=john_2_3, pic=pic_3.jpg}, {id=4, sex=0, name=john_2_4, pic=pic_4.jpg}]}
6)最后把main方法里面的代碼放到web層,只需要定義一個query參數(shù),很容易就把查詢服務(wù)搭建好了,dataFetcher 里面還是調(diào)用原來的查詢接口
7)引入maven依賴
<dependency> <groupId>com.graphql-java</groupId> <artifactId>graphql-java</artifactId> <version>2.0.0</version> </dependency>
關(guān)于graphql查詢什么定義,看看這個或許對你有幫助
json
{
id=6,
sex=0,
name="Name_6",
pic="pic_6.jpg"
}
query
{
id,
sex,
name,
pic
}
后面那部分,其實就是json字符串,去掉=和value的結(jié)果,還是可讀的
結(jié)語
graphql 帶了一種全新的思維方式,可以簡化web api的開發(fā),由客戶端指定需要什么數(shù)據(jù),服務(wù)端返回什么數(shù)據(jù),減少不必要的流量傳輸,對移動端友好,還提供多種數(shù)據(jù)聚合查詢,多個查詢只是用一個請求,既滿足api最小粒度,又滿足前端需要,減少請求,提高性能。
感覺以后會朝這方面去發(fā)展,大趨所驅(qū)。
相關(guān)文章
shiro與spring集成基礎(chǔ)Hello案例詳解
這篇文章主要介紹了shiro與spring集成基礎(chǔ)Hello案例詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-11-11

