小洋的前端记事本(NO.5):GraphQL 小实践

简介: **这里记录着小洋童鞋在前端道路上的所思所想所感。** >生命在于折腾,不在折腾中崩溃,就在折腾中涅槃。 ------ [GraphQL| A query language for your API](http://graphql.org/),诞生于移动快速迭代时代,理念是给客户端提供类似”数据库操作“的能力,使其能够从服务接口这个大的“大数据库”中去筛选或修改想要的数据,用以满

这里记录着小洋童鞋在前端道路上的所思所想所感。

生命在于折腾,不在折腾中崩溃,就在折腾中涅槃。


GraphQL| A query language for your API,诞生于移动快速迭代时代,理念是给客户端提供类似”数据库操作“的能力,使其能够从服务接口这个大的“大数据库”中去筛选或修改想要的数据,用以满足前端数据的个性化需求,既保证了多样性,又控制了接口数量,期望更好的分类维护接口。GraphQL 服务器有很多的实现,Node.js 当然也不例外,下面我们就一起折腾折腾 GraphQL 吧 : )

GraphQL Node.js Server

准备

首先给大家简单的介绍一下 GraphQL 特性,GraphQL 其实是一种和后端交换数据的协议,像通过 SQL 语言可以操作数据库一样,GraphQL 通过 Schema 的方式来控制数据,并约定提供以下能力:

  • query 查询数据 详见
  • mutation 特殊查询,可以看做为修改数据 详见
  • introspection 检查数据的支持情况(下文简称”自省”) 详见

可以说很多数据交换的情景都可以使用 GraphQL 的概念,我们就先基于 Express 来快速打造一个 http 服务器 ( GraphQL 官方并没有说只限于 http) 用最简单的方式体验一下 GraphQL,我们的项目起始于 package.json:

{
  "name": "graphql-start",
  "description": "简单体验一下 GraphQL ",
  "private": true, 
  "dependencies": {
    "babel-register": "^6.18.0", // ES6 你懂的
    "babel-preset-es2015": "^6.18.0", // ES6 你懂的
    "express": "^4.14.0", // 快速搭一个服务器
    "body-parser": "^1.15.2", // Express 中间件,获取 GraphQL 请求用的
    "graphql": "^0.8.2" // 我们的主角,处理 GraphQL 请求用的
  }
}

$npm install 以后我们就可以愉快的玩耍了,首先搭好程序入口,本地服务以及处理 GraphQL 请求的 handler:

/* 入口:index.js */
require('babel-register')({
    presets: [ 'es2015' ]
});
require('./server.js');

/* 本地服务:server.js */
// base
import express from 'express';
import {graphql} from 'graphql';
import bodyParser from 'body-parser';

// 组织服务端识别的 schema
import schema from './schema';

let app = express();
let PORT = 3000;

// 用 text 的方式解析 request doby,形象起见我们设置一个特殊的 Content-Type
app.use(bodyParser.text({type: 'application/graphql'}));

// 创建 "/graphql" 的路由(可以看做一个API)接收 GraphQL 请求,使用 GET ? 特殊字符,url链接长度不考虑啦?
app.post('/graphql', (req, res) => {
    // 组织服务端识别的 schema, 并处理 GraphQL 请求
    graphql(schema, req.body).then((result) => {
        res.send(JSON.stringify(result, null, 2));
    })
});

let server = app.listen(PORT, function () {
    let host = server.address().address;
    let port = server.address().port;

    console.log('GraphQL listening at http://127.0.0.1', host, port);
});

/* schema 实例:schema.js */
import {GraphQLSchema, GraphQLInt, GraphQLString, GraphQLBoolean, GraphQLObjectType, GraphQLList} from 'graphql';

// 创建一个GraphQLSchema实例,它提供了一个配置,下面主要来写我们的顶级键 query 和 mutation 
let schema = new GraphQLSchema({
  query: ... ,
  mutation: ...
});

export default schema;

上述代码创建了入口 index.js 、本地服务 server.js 以及 schema 实例 schema.js ,至此我们的 GraphQL 服务的骨架就出来了,下面我们只要来把 schema.js 填充完整就可以体验 GraphQL 啦~

GraphQL Schema

以下所有代码均在 schema.js 中,开始前可以先了解一下 graphql-js

g1.png

我们可以按照上述描述数据筛选数据然后获取数据的顺序生成测试代码感受一下 GraphQL 的魅力吧~

目标数据

首先我们先创建一个模拟数据源 todo list:

let TODOs = [
    {
        "id": 111,
        "title": "Read emails",
        "completed": false
    },
    {
        "id": 222,
        "title": "Buy orange",
        "completed": true
    }
];

todo list 是一个元素为 todo object 的数组,由于我们要给客户端提供筛选及自省的能力,所以我们要通过 GraphQL 的 type 来描述所有组织结构,首先我们来描述元素 todo object

// 申明 todo object 的结构及组成
let TodoType = new GraphQLObjectType({
    name: 'todo', 
    fields: function () {
        return {
            id: {
                type: GraphQLInt, // Int 型
                description: "todo object's id"
            },
            title: {
                type: GraphQLString, // String 型
                description: "todo object's title"
            },
            completed: {
                type: GraphQLBoolean, // Boolean 值
                description: "todo object's status"
            }
        }
    }
});

那么 todo list 显而易见就是 new GraphQLList(TodoType) 啦,即元素为 TodoType 的数组。

Schema

有了数据源我们就可以愉快的通过 GraphQL Schema 来取数据啦,我们的第一个 Schema 就先为获取整个 todo list 吧,先在服务器上注册一个处理 GraphQL 请求的方法:

let schema = new GraphQLSchema({
    query: new GraphQLObjectType({
        name: 'getTODOs', // query name
        description: 'The TODOs!',
        fields: function () {
            return {
                TODOs: { // 返回结果 {TODOs: [...]}
                    type: new GraphQLList(TodoType),
                    description: 'The TODOs List!',
                    resolve: function () {
                        // 返回 todo list
                        return TODOs;
                    }
                }
            }
        }
    })
    //, mutation: ...
});

之前已经创建了一个端口为3000,路由为 /graphql 的 GraphQL API,至此终于可以 $node index.js 开始我们的 GraphQL 之旅啦~

query

比如我们可以通过下面的 query 来获取 TODOs 下的 title ,由于 TODOs 是数组所以结果对应也是数组:

g3.png

注:通过控制台,Postman等均可构造 POST 请求验证结果,本文所有请求都通过 curl 来构造(包括添加 ContentType 请求头及 data)。

introspection

在介绍自省之前,我们可以先了解一下域,每个 GraphQL 根域都有 __schema 域,这个域有一个子域叫 queryType。我们可以通过查询这些域来了解 GraphQL 服务器支持那些查询,按照 Object 的方式理解其实就是通过对象的属性可以了解其结构及内容

g4.png

看到上面的结果大家应该都有感觉了,其实 GraphQL 请求的成功与否与接口的 Schema 设计结构息息相关,所有的数据结构查询都是按协商好的规则进行的,还记得我们声明的 GraphQLSchema 是怎么写的吗?

{
    name: 'getTODOs',
    description: 'The TODOs!',
    fields: function () {
        return {
            TODOs: {
                type: new GraphQLList(TodoType),
                description: 'The TODOs List!'
                ...
            }
        }
    }
}

发现结果是一一对应的了吧,哈哈~ 可以记住这个“万能查询”,同时 GraphQL 也提供了__Type, __TypeKind, __Field, __InputValue等等的关键字来检测接口所支持查询的属性,比如我们现在已经知道了“/garphql”这个 API 会返回一个对象数组,那我想知道他包含的对象是什么结构咋变咧?

g5.png

没错我们可以通过限定 __Type 去查已经注册的 "todo" 对象来自省接口,还有很多关键字方法都可以达到上述的效果本文就不一一列举了。细心的同学发现上面 __type(name: todo) 的用法了吧,同样也可以用到我们的代码中来查询具体某一条的 todo 对象:

let schema = new GraphQLSchema({
    query: new GraphQLObjectType({
        name: 'getTODO',
        fields: function () {
            return {
                todo: {
                    type: TodoType, // 这次我们只返回一个 todo 对象
                    description: 'The TODO!',
                    args: {
                        id: {
                            type: GraphQLInt,
                            required: true 
                        }
                    },
                    resolve: function (root, param, context) {
                        console.log(param.id);
                          // 这里就可以放和数据库的交互了,本例就简单的返回第一个 todo 对象
                        return TODOs[0];
                    }
                }
            }
        }
    })
    //, mutation: ...
});

g6.png

mutation

有查询数据当然也有修改数据,mutation 查询和普通查询请求(query)的重要区别在于 mutation 操作是按照顺序执行的,即多次修改数据后再查询,其返回一定是最后一次修改后的结果:

let schema = new GraphQLSchema({
    // query: ... ,
    mutation: new GraphQLObjectType({
        name: 'updateTODO',
        fields: {
            updateTODO: {
                type: TodoType,
                description: 'Update the TODO status',
                args: {
                    id: {
                        type: GraphQLInt,
                        required: true
                    }
                },
                resolve: function (root, param, context) {
                   // 这里就可以放和数据库的交互了,本例就简单的更新 completed 属性
                      console.log(param.id);
                    TODOs[0].completed = true;
                    return TODOs[0];
                }
            }
        }
    })
});

g7.png

调试

GraphiQL 可以快速联想搜索接口,并提供了非常强大的自省能力,是很好的提高 GraphQL 开发效率的工具

g2.png

总结

正如官方所说 GraphQL 是一个查询语言,而且目前还未完成,未来也可能会有更多更大的变动,但已经拓宽了我们的思路,让我们看到了更多的可能性。目前 GraphQL 利好主要是在于前端的开发效率,落地时需要服务端的全力配合,也存在着一定的安全风险(暴力破解接口能力等),最大的性能瓶颈可能来自于数据库查询,但究竟好不好用,关键是看你怎么用了,对吧? 我们仍在前行,阳光总在风雨后。

目录
相关文章
|
2月前
|
缓存 前端开发 JavaScript
利用代码分割优化前端性能:策略与实践
在现代Web开发中,代码分割是提升页面加载性能的有效手段。本文介绍代码分割的概念、重要性及其实现策略,包括动态导入、路由分割等方法,并探讨在React、Vue、Angular等前端框架中的具体应用。
|
1月前
|
编解码 前端开发 开发者
探索无界:前端开发中的响应式设计深度实践与思考###
本文将带你领略响应式设计的精髓,一种超越传统页面布局限制的设计策略,它要求开发者以灵活多变的思维,打造能够无缝适应各种设备与屏幕尺寸的Web体验。通过深入浅出的讲解、实际案例分析以及技术实现细节的探讨,本文目的是激发读者对于响应式设计深层次的理解与兴趣,鼓励在实际应用中不断创新与优化。 ###
78 10
|
2月前
|
编解码 前端开发 开发者
前端开发中的响应式设计实践
前端开发中的响应式设计实践
|
2月前
|
编解码 前端开发 UED
探索无界:前端开发中的响应式设计深度解析与实践####
【10月更文挑战第29天】 本文深入探讨了响应式设计的核心理念,即通过灵活的布局、媒体查询及弹性图片等技术手段,使网站能够在不同设备上提供一致且优质的用户体验。不同于传统摘要概述,本文将以一次具体项目实践为引,逐步剖析响应式设计的关键技术点,分享实战经验与避坑指南,旨在为前端开发者提供一套实用的响应式设计方法论。 ####
71 4
|
2月前
|
编解码 前端开发 UED
探索无界:前端开发中的响应式设计哲学与实践####
本文不拘泥于传统摘要的框架,而是以一种对话的方式,引领读者踏入响应式设计的奇妙世界。想象一下,互联网如同一片浩瀚的海洋,而网页则是航行其中的船只。在这片不断变化的海域中,如何让我们的“船只”既稳固又灵活地适应各种屏幕尺寸和设备?这正是响应式设计的魅力所在。通过深入浅出的探讨,我们将一同揭开它背后的哲学思想与实战技巧,让你的网页在任何设备上都能展现出最佳姿态。 ####
31 0
|
2月前
|
前端开发 JavaScript API
现代前端框架中的响应式编程实践
现代前端框架中的响应式编程实践
43 0
|
3月前
|
人工智能 资源调度 数据可视化
【AI应用落地实战】智能文档处理本地部署——可视化文档解析前端TextIn ParseX实践
2024长沙·中国1024程序员节以“智能应用新生态”为主题,吸引了众多技术大咖。合合信息展示了“智能文档处理百宝箱”的三大工具:可视化文档解析前端TextIn ParseX、向量化acge-embedding模型和文档解析测评工具markdown_tester,助力智能文档处理与知识管理。
|
2月前
|
缓存 监控 前端开发
前端开发中的性能优化:策略与实践
前端开发中的性能优化:策略与实践
|
2月前
|
编解码 前端开发 UED
前端开发中的响应式设计实践
前端开发中的响应式设计实践
45 0
|
3月前
|
JavaScript 前端开发 Docker
拿下奇怪的前端报错(二):nvm不可用报错`GLIBC_2.27‘‘GLIBCXX_3.4.20‘not Found?+ 使用docker构建多个前端项目实践
本文介绍了在多版本Node.js环境中使用nvm进行版本管理和遇到的问题,以及通过Docker化构建流程来解决兼容性问题的方法。文中详细描述了构建Docker镜像、启动临时容器复制构建产物的具体步骤,有效解决了不同项目对Node.js版本的不同需求。
138 0
下一篇
开通oss服务