写点什么

为什么说 GraphQL 可以取代 REST API?

  • 2019-01-21
  • 本文字数:9058 字

    阅读完需:约 30 分钟

为什么说GraphQL可以取代REST API?

几年前,我在 DocuSign 带领了一个开发团队,任务是重写一个有数千万个用户在使用的 Web 应用程序。当时还没有可以支持前端的 API,因为从一开始,Web 应用程序就是一个.NET 大单体。西雅图的 API 团队在将拆分单体,并逐步暴露出 RESTful API。这个 API 团队由两名工程师组成,发布周期为一个月,而我们在旧金山的前端团队每周都会发布新版本。


API 团队的发布周期太长,因为很多(几乎所有)功能都必须进行手动测试,这是可以理解的。它毕竟是一个单体,而且没有适当的自动化测试——如果他们修改了一个地方,不知道在应用程序的其他地方会出现什么问题。


我记得有一次,我们的前端团队面临为某大会交付新版本的压力,但我们忘记跟进一个重要的 API 变更,这个变更未被包含在即将发布的 API 版本中。我们要么一直等待,直到错过截止日期,要么有人愿意放弃优先权,以便让我们的变更包括在即将发布的版本中。所幸的是,这个变更最后被包含在新版本中,我们也及时发布了新的前端版本。我真的希望当时我们已经使用了 GraphQL,因为它可以消除对外部团队及其发布周期的重度依赖。


在这篇文章中,我将介绍 GraphQL 的优势,以及为什么它会变得如此受欢迎。


很多公司已经在内部从 RESTful 转向了 GraphQL API:IBM、Twitter、Walmart Labs、纽约时报、Intuit、Coursera,等等。


其他一些公司不仅是在内部而且还将外部 API 也转为 GraphQL:AWS、Yelp、GitHub、Facebook 和 Shopify,等等。GitHub 甚至打算停止使用 REST API,他们的 v4 版本只使用 GraphQL。


GraphQL 究竟是一个炒作流行语还是真正会带来一场变革?有趣的是,我之前列出的大多数从 GraphQL 获益的公司都有以下这些共同点。


  • 他们拥有包括移动端在内的多个客户端;

  • 他们正在转向或者已经采用了微服务架构;

  • 他们的遗留 REST API 数量暴增,变得十分复杂;

  • 他们希望消除客户端团队对 API 团队的依赖;

  • 他们注重良好的 API 文档和开发者体验。


GitHub 工程团队表明了他们的动机:


“GraphQL 弥合了发布的内容与可以使用的内容之间的差距。我们真的很期待能够同时发布它们。GraphQL 代表了 API 开发的巨大飞跃。类型安全、内省、生成文档和可预测的响应都为我们平台的维护者和消费者带来了好处。我们期待着由 GraphQL 提供支持的平台进入新时代,也希望你们也这样做!”


GraphQL 加速了开发速度,提升了开发者体验,并提供了更好的工具。我并不是说这绝对是这样的,但我会尽力说明 GraphQL 与 REST 之间的争论点及其原因。

超级数据聚合器

我是 Indeed(世界排名第一的求职网站)的软件工程负责人,所以让我们先来看看 Indeed.com 的主页和职位查询结果页面。它们分别发出了 10 和 11 个 XHR 请求。



需要注意的是,在 REST 中使用 POST 进行页面浏览并不是很“正规”。



以下是其中的一些调用:



在使用 GraphQL 时,上面的这些请求可以被包含在单个查询和单个请求中。


query HomePage {  getConversationCount(...) {     ...  }  jobdescs(...) {     ...  }  vjslog(...) {     ...  }  preccount(...) {  }  jobalert(...) {  }  count(...) {  }}
复制代码


响应结果可能是这样的:


{  "data": {    "getConversationCount": [      {        ...      }    ],    "vjslog": [...],    "preccount": [...],      "jobalert": [...],    "count": {}  },  "errors": []}
复制代码


通常,单个调用比多个调用更方便、更有效,因为它需要更少的代码和更少的网络开销。来自 PayPal 过程团队的开发体验还证实,很多 UI 工作实际上不是 UI 工作,而是其他任务,例如前端和后端之间的通信:


“我们发现,UI 开发人员实际用于构建 UI 的时间不到三分之一,剩下的时间用于确定在何处以及如何获取数据、过滤/映射数据以及编排 API 调用,还有一些用于构建和部署。”


需要注意的是,有实时使多个请求也是有必要的,例如多个单独的请求可以快速且异步独立地获取不同的数据,如果采用了微服务架构,它们会增加部署灵活性,而且它们的故障点是多个,而不是一个。


此外,如果页面是由多个团队开发的,GraphQL 提供了一个功能,可以将查询分解称为片段。稍后我们将详细介绍这方面的内容。


从更大的角度来看,GraphQL API 的主要应用场景是 API 网关,在客户端和服务之间提供了一个抽象层。



微服务架构很好,但也存在一些问题,GraphQL 可以用来解决这些问题。以下是来自 IBM 在微服务架构中使用 GraphQL 的经验:


“总的来说,GraphQL 微服务的开发和部署都非常快。他们 5 月份开始开发,7 月份就进入了生产环境。因为他们不需要征得许可,直接开干。他强烈推荐这个方案,比开会讨论好太多了。”


接下来,让我们逐一讨论 GraphQL 的每一个好处。

提高开发速度

首先,GraphQL 有助于减少发出的请求数。通过单个调用来获取所需的数据比使用多个请求要容易得多。从工程师的角度来看,这加快了开发速度。后面我会解释更多有关为什么会提升开发速度的原因,但现在我想先说明另一个问题。


后端和客户端团队需要通过密切合作来定义 API、测试它们,并做出更改。前端、移动、物联网(例如 Alexa)等客户端团队不断迭代功能,并尝试使用新的 UX 和设计。他们的数据需求经常发生变化,后端团队必须跟上他们的节奏。如果客户端和后端代码由同一团队负责,那么问题就没那么严重了。Indeed 的大多数工程团队都是由全栈工程师组成,但并非全部都是这样。对于非全栈团队,客户端团队经常因为依赖了后端团队开发速度受到影响。


当我转到 Job Seeker API 团队时,移动团队开始我们的开发进度。我们之间有很多关于参数、响应字段和测试的事情需要沟通。


在使用了 GraphQL 之后,客户端工程师就可以完全控制前端,不需要依赖任何人,因为他们可以告诉后端他们需要什么以及响应结构应该是怎样的。他们使用了 GraphQL 查询,它们会告诉后端 API 应该要提供哪些数据。


客户端工程师不需要花时间让后端 API 团队添加或修改某些内容。GraphQL 具有自文档的特点,所以可以节省一些用于查找文档以便了解如何使用 API 的时间。我相信大多数人曾经在找出确切的请求参数方面浪费了很多时间。GraphQL 协议本身及其社区在文档方面为我们提供了一些有用的工具。在某些情况下,可以从模式自动生成文档。其他时候,只需使用 GraphiQL Web 界面就足以编写一个查询。


来自纽约时报的工程师表示,他们在转到 GraphQL 和 Relay 之后,在做出变更时不需要改太多的东西:


“当我们想要更新所有产品的设计时,不再需要修改多个代码库。这就是我们想要的。我们认为 Relay 和 GraphQL 是帮助我们实现这个伟大目标的完美工具。”


当一家公司已经拥有大量 GraphQL API,然后有人想出了一个新的产品创意,这也是我最喜欢 GraphQL 的应用场景。使用已有的 GraphQL API 实现原型比调用各种 REST 端点(将提供太少或太多的数据)或为新应用程序构建新的 REST API 要快得多。


开发速度的提升与开发者体验的提升密切相关。

提升开发者体验

GraphQL 提供了更好的开发者体验(DX),开发者将花更少的时间思考如何获取数据。在使用 Apollo 时,他们只需要在 UI 中声明数据。数据和 UI 放在一起,阅读代码和编写代码都变得更方便。


通常,在开发 UI 时需要在 UI 模板、客户端代码和 UI 样式之间跳转。GraphQL 允许工程师在客户端开发 UI,减少摩擦,因为工程师在添加或修改代码时无需在文件之间切换。如果你熟悉 React,这里有一个很好的比喻:GraphQL 之于数据,就像 React 之于 UI。


下面是一个简单的示例,UI 中直接包含了属性名称launch.namelaunch.rocket.name


const GET_LAUNCHES = gql`  query launchList($after: String) {    launches(after: $after) {      launches {        id        name        isBooked        rocket {          id          name        }      }    }  }`;
export default function Launches() { return ( <Query query={GET_LAUNCHES}> {({ data, loading, error }) => { if (loading) return <Loading />; if (error) return <p>ERROR</p>;
return ( <div> {data.launches.launches.map(launch => ( <div key={launch.id} >{launch.name}<br/> Rocket: {launch.rocket.name} </div> ))} </div> ); }} </Query> );};
复制代码


使用这种方法,可以非常容易地修改或向 UI 或查询(gql)添加新字段。React 组件的可移植性更强了,因为它们描述了所需的所有数据。


如前所述, GraphQL 提供了更好的文档,而且还有一个叫作 GraphiQL 的 IDE:



前端工程师很喜欢 GraphiQL,下面引用 Indeed 的一位高级工程师说过的话:


“我认为开发体验中最好的部分是能够使用 GraphiQL。对我来说,与典型的 API 文档相比,这是一种编写查询更有效的辅助方法”。


GraphQL 的另一个很棒的功能是片段,因为它允许我们在更高的组件层面重用查询。


这些功能改善了开发者体验,让开发人员更快乐,更不容易出现 JavaScript 疲劳。

提升性能

工程师并不是唯一从 GraphQL 中受益的人。用户也会从中受益,因为应用程序的性能获得了提升(可以感知到的):


1.减少了有效载荷(客户端只需要必要的东西);


2.多个请求合并为一个请求可减少网络开销;


3.使用工具可以更轻松地实现客户端缓存和后端批处理和后端缓存;


4.预取;


5.更快的 UI 更新。


PayPal 使用 GraphQL 重新设计了他们的结账流程。下面是来自用户的反馈:


“REST 的原则并没有为 Web 和移动应用及其用户的需求考虑,这个在结账优化交易中体现得尤为明显。用户希望能够尽快完成结账,如果应用程序使用了很多原子 REST API,就需要在客户端和服务器之间进行多次往返以获取数据。我们的结账每次往返网络时间至少需要 700 毫秒,这还不包括服务器处理请求的时间。每次往返都会导致渲染变慢,用户体验不好,结算转换率也会降低。”


性能改进中有一项是“多个请求组合成一个请求可以减少网络开销”。对于 HTTP/1 而言,这是非常正确的,因为它没有 HTTP/2 那样的多路复用机制。但尽管 HTTP/2 提供的多路复用机制有助于优化单独的请求,但它对于图遍历(获取相关或嵌套对象)并没有实际帮助。让我们来看一看 REST 和 GraphQL 是如何处理嵌套对象和其他复杂请求的。

标准化和简化复杂的 API

通常,客户端会发出复杂的请求来获取有序、排好序、被过滤过的数据或子集(用于分页),或者请求嵌套对象。GraphQL 支持嵌套数据和其他难以使用标准 REST API 资源(也叫端点或路由)实现的查询。


例如,我们假设有三种资源:用户、订阅和简历。工程师需要按顺序进行两次单独的调用(这会降低性能)来获取一个用户简历,首先需要通过调用获取用户资源,拿到简历 ID,然后再使用简历 ID 来获取简历数据。对于订阅来说也是一样的。


1.GET /users/123:响应中包含了简历 ID 和工作岗位通知订阅的 ID 清单;


2.GET /resumes/ABC:响应中包含了简历文本——依赖第一个请求;


3.GET /subscriptions/XYZ:响应中包含了工作岗位通知的内容和地址——依赖第一个请求。


上面的示例很糟糕,原因有很多:客户端可能会获得太多数据,并且必须等待相关的请求完成了以后才能继续。此外,客户端需要实现如何获取子资源(例如建立或订阅)和过滤。


想象一下,一个客户端可能只需要第一个订阅的内容和地址以及简历中的当前职位,另一个客户端可能需要所有订阅和整个简历列表。所以,如果使用 REST API,对第一个客户端来说有点不划算。


另一个例子:用户表里可能会有用户的名字和姓氏、电子邮件、简历、地址、电话、社会保障号、密码(当然是经过混淆的)和其他私人信息。并非每个客户端都需要所有字段,有些应用程序可能只需要用户电子邮件,所以向这些应用程序发送社会保障号等信息就不太安全。


当然,为每个客户端创建不同的端点也是不可行的,例如/api/v1/users 和/api/v1/usersMobile。事实上,各种客户端通常都有不同的数据需求:/api/v1/userPublic、/api/v1/userByName、/api/v1/usersForAdmin,如果这样的话,端点会呈指数级增长。


GraphQL 允许客户要求 API 发送他们想要的字段,这将使后端工作变得更加容易:/api/gql——所有客户端只需要这个端点。


注意:对于 REST 和 GraphQL,后端都需要使用访问控制级别。


或者可以使用旧 REST 来实现 GraphQL 的很多功能。但是这样要付出什么代价?后端可以支持复杂的 RESTful 请求,这样客户端就可以使用字段和嵌套对象进行调用:


GET /users/?fields=name,address&include=resumes,subscriptions
复制代码


上面的请求将比使用多个 REST 请求更好,但它不是标准化的,不受客户端库支持,而且这样的代码也更难编写和维护。对于相对复杂的 API,工程师需要在查询中使用自己的查询字符串参数约定,最终得到类似 GraphQL 的东西。既然 GraphQL 已经提供了标准和库,为什么还要基于 REST 设计自己的查询约定呢?


将复杂的 REST 端点与以下的 GraphQL 嵌套查询进行对比,嵌套查询使用了更多的过滤条件,例如“只要给我前 X 个对象”和“按时间按升序排列”(可以添加无限制的过滤选项):


{    user (id: 123) {        id        firstName        lastName        address {            city            country            zip        }        resumes (first: 1, orderBy: time_ASC) {            text            title            blob            time          }          subscriptions(first: 10) {            what            where            time        }    }}}
复制代码


在使用 GraphQL 时,我们可以在查询中保留嵌套对象,对于每个对象,我们将精确地获得我们需要的数据,不多也不少。


响应消息的数据格式反映了请求查询的结构,如下所示:


{    "data": {        "user": {            "id": 123,            "firstName": "Azat",            "lastName": "Mardan",            "address": {                "city": "San Francisco",                "country": "US",                "zip": "94105"            },            "resumes" [                  {                    "text": "some text here...",                    "title": "My Resume",                    "blob": "<BLOB>",                    "time": "2018-11-13T21:23:16.000Z"                  },            ],              "subscriptions": [ ]        },    "errors": []    }
复制代码


相比复杂的 REST 端点,使用 GraphQL 的另一个好处是提高了安全性。这是因为 URL 经常会被记录下来,而 RESTful GET 端点依赖于查询字符串(是 URL 的一部分)。这可能会暴露敏感数据,所以 RESTful GET 请求的安全性低于 GraphQL 的 POST 请求。我打赌这就是为什么 Indeed 主页会使用 POST 发出“阅读”页面请求。


使用 GraphQL 可有更容易地实现分页等关键功能,这要归功于查询以及 BaaS 提供商提供的标准,以及后端的实现和客户端库使用的标准。

改进的安全性、强类型和验证

GraphQL 的 schema 与语言无关。对前面的示例进行扩展,我们可以在 schema 中定义 Address 类型:


type Address {    city: String!    country: String!    zip: Int}
复制代码


String 和 Int 是标量类型,!表示字段不可为空。


schema 验证是 GraphQL 规范的一部分,因此像这样的查询将返回错误,因为 name 和 phone 不是 Address 对象的字段:


{    user (id: 123) {        address {            name            phone        }    }}
复制代码


我们可以使用我们的类型构建复杂的 GraphQL schema。例如,用户类型可能会使用我们的地址、简历和订阅类型,如下所示:


type User {    id: ID!    firstName: String!    lastName: String!    address: Address!    resumes: [Resume]     subscriptions: [Subscription]}
复制代码


Indeed 的大量对象和类型都是使用 ProtoBuf 定义的。类型化数据并不是什么新鲜事物,而且类型数据的好处也是众所周知。与发明新的 JSON 类型标准相比,GraphQL 的优点在于已经存在可以从 ProtoBuf 自动换换到 GraphQL 的库。即使其中一个库(rejoiner)不能用,也可以开发自己的转换器。


GraphQL 提供了比 JSON RESTful API 更强的安全性,主要有两个原因:强类型 schema(例如数据验证和无 SQL 注入)以及精确定义客户端所需数据的能力(不会无意泄漏数据)。


静态验证是另一个优势,可以帮助工程师节省时间,并在进行重构时提升工程师的信心。诸如eslint-plugin-graphql之类的工具可以让工程师知道后端发生的变化,并让后端工程师确保不会破坏客户端代码。


保持前端和后端之间的契约是非常重要的。在使用 REST API 时,我们要小心不要破坏了客户端代码,因为客户端无法控制响应消息。相反,GraphQL 为客户端提供了控制,GraphQL 可以频繁更新,而不会因为引入了新类型造成重大变更。因为使用了 schema,所以 GraphQL 是一种无版本的 API。

GraphQL 的实现

在选择实现 GraphQL API 的平台时,Node 是一个候选项,因为最初 GraphQL 用于 Web 应用程序和前端,而 Node 是开发 Web 应用程序的首选,因为它是基于 JavaScript 的。使用 Node 可以非常容易地实现 GraphQL(假设提供了 schema)。事实上,使用 Express 或 Koa 来实现只需要几行代码:


const Koa = require('koa');const Router = require('koa-router'); // koa-router@7.xconst graphqlHTTP = require('koa-graphql');
const app = new Koa();const router = new Router();
router.all('/graphql', graphqlHTTP({ schema: schema, graphiql: true}));
app.use(router.routes()).use(router.allowedMethods());
复制代码


schema 是使用 npm 的 graphql 中的类型来定义的。Query 和 Mutation 是特殊的 schema 类型。


GraphQL API 的大部分实现都在于 schema 和解析器。解析器可以包含任意代码,但最常见的是以下五个主要类别:


  • 调用 Thrift、gRPC 或其他 RPC 服务;

  • 调用 HTTP REST API(当优先事项不是重写现有 REST API 时);

  • 直接调用数据存储;

  • 调用其他 GraphQL schema 查询或服务;

  • 调用外部 API。


这里有一个示例


Node 很棒,但在 Indeed,我们主要使用 Java。包括 Java 在内的很多语言都支持 GraphQL,例如https://github.com/graphql-gohttps://github.com/graphql-python


由于 Indeed 主要使用了 Java,因此这里给出一个使用 graphql-java 的 Java GraphQL 示例,完整代码位于这里。它定义了/graphql 端点:


import com.coxautodev.graphql.tools.SchemaParser;import javax.servlet.annotation.WebServlet;import graphql.servlet.SimpleGraphQLServlet;
@WebServlet(urlPatterns = "/graphql")public class GraphQLEndpoint extends SimpleGraphQLServlet {
public GraphQLEndpoint() { super(SchemaParser.newParser() .file("schema.graphqls") //parse the schema file created earlier .build() .makeExecutableSchema()); }}
复制代码


GraphQL 的 schema 使用 POJO 来定义。GraphQL 端点类使用了 LinkRepository POJO。解析器包含了操作的(例如获取链接)实际代码:


@WebServlet(urlPatterns = "/graphql")public class GraphQLEndpoint extends SimpleGraphQLServlet {
public GraphQLEndpoint() { super(buildSchema()); }
private static GraphQLSchema buildSchema() { LinkRepository linkRepository = new LinkRepository(); return SchemaParser.newParser() .file("schema.graphqls") .resolvers(new Query(linkRepository)) .build() .makeExecutableSchema(); }}
复制代码


在很多情况下,GraphQL 的 schema 可以从其他类型的 schema 自动生成,例如 gRPC、Boxcar、ProtoBuf 或 ORM/ODM。


GraphQL 不一定需要客户端。一个简单的 GraphQL 请求就是一个常规的 POST HTTP 请求,其中包含了查询内容。我们可以使用任意的 HTTP 代理库(如 CURL、axios、fetch、superagent 等)来生成请求。例如,在终端中使用 curl 发送请求:


curl \  -X POST \  -H "Content-Type: application/json" \  --data '{ "query": "{ posts { title } }" }' \  https://1jzxrj179.lp.gql.zone/graphql
复制代码


以下代码可以在任意一个现代浏览器(为了避免 CORS,请访问launchpad.graphql.com)中运行。


fetch('https://1jzxrj179.lp.gql.zone/graphql', {  method: 'POST',  headers: { 'Content-Type': 'application/json' },  body: JSON.stringify({ query: '{ posts { title } }' }),})  .then(res => res.json())  .then(res => console.log(res.data));
复制代码


虽然构建 GraphQL 请求很容易,但是还需要实现很多其他东西,比如缓存,因为缓存可以极大地改善用户体验。构建客户端缓存不是那么容易,所幸的是,Apollo 和 Relay Modern 等提供了开箱即用的客户端缓存。

什么时候不该使用 GraphQL?

当然,完美的解决方案是不存在的(尽管 GraphQL 接近完美),还有一些问题需要注意,例如:


1.它有单点故障吗?


2.它可以扩展吗?


3.谁在使用 GraphQL?


最后,以下列出了我们自己的有关 GraphQL 可能不是一个好选择的主要原因:


  • 当客户端的需求很简单时:如果你的 API 很简单,例如/users/resumes/123,那么 GraphQL 就显得有点重了;

  • 为了加快加载速度使用了异步资源加载;

  • 在开发新产品时使用新的 API,而不是基于已有的 API;

  • 不打算向公众公开 API;

  • 不需要更改 UI 和其他客户端;

  • 产品开发不活跃;

  • 使用了其他一些 JSON schema 或序列化格式。

总结

GraphQL 是一种协议和一种查询语言。GraphQL API 可以直接访问数据存储,但在大多数情况下,GraphQL API 是一个数据聚合器和一个抽象层,一个可以提升开发速度、减少维护工作并让开发人员更快乐的层。因此,GraphQL 比公共 API 更有意义。很多公司开始采用 GraphQL。IBM、PayPal 和 GitHub 声称在使用 GraphQL 方面取得了巨大的成功。如果 GraphQL 很有前途,我们现在是否可以停止构建过时且笨重的 REST API,并拥抱 GraphQL?


英文原文:https://webapplog.com/graphql/


更多内容,请关注前端之巅(ID:frontshow)



2019-01-21 09:5727415
用户头像

发布了 731 篇内容, 共 454.3 次阅读, 收获喜欢 2003 次。

关注

评论 13 条评论

发布
用户头像
说是前后端不用再沟通了,我怎么觉得这样反而前后端必须沟通才能使用了呢,要不我怎么知道后端资源的每个字段和关系呢,如果用restful,至少通过url的能读懂这个接口到底是做什么的,看下返回结果就知道资源结构,通过include 可以实现一个接口调用多个关联资源。而且restful后端只需要关注某个资源和联系,感觉用了这个都耦合到一起了。
2019-12-19 16:12
回复
用户头像
GraphQL 还是可以的。
虽然本质上都是资源的获取和操作,但是在一定程度可以提高前端的开发效率。
当然,得让后端先把模型关联好。
有点类似前端直接写SQL传给后端,但是更安全。
且关联模型由后端还是控制。
当然,门槛相对于REST来说,还是有点颇高,期待各大厂家能把多讲讲实战经验以及力所能及的开源,有利于行业的发展。
2019-05-13 15:25
回复
用户头像
REST本意是资源获取,没有像RPC一样有服务调用的意味。GraphQL定义为一种查询语言,像另一位同学说的“SQL前移”,本质上还是资源获取,也没有看到服务调用的意味。个人拙见,两者没有什么本质变化,只是形式变了而已,至于API稳定的问题,那不是技术导致的,而是设计问题,就像现实中拿REST实现服务接口不在少数吧。
2019-05-04 20:43
回复
用户头像
当年用ajax分布加载不就是为了让页面先出来,数据分步填入么,GraphQL这样等统一返回,一个大页面要等我后台20个sql查完了再一起出来?感觉回到了10多年前的老路啊
2019-01-29 11:55
回复
用户头像
GraphQL的意义是将后台数据模型,以一种安全的方式暴露给前端进行定制化查询,模型统一后可以减少后台的各种定制查询接口。
2019-01-29 11:02
回复
用户头像
直接拿到关联数据的解决方案, 而不是一直传各种主键
2019-01-27 13:08
回复
用户头像
令人激动,跃跃欲试
2019-01-27 11:08
回复
用户头像
没看出优势在哪里
2019-01-21 15:55
回复
用户头像
好, 假设真的REST API数量暴增了, 然后通过GraphQL能解决吗? GraphQL的API数量倒是减少了, 但数据中携带的元数据的复杂性是不是也会暴增了? Essential Complex不会因为实现方式的改变而减少, 只会从一个地方转移到另外一个地方。 尝试通过改变实现方式来减少Essential Complex,注定会徒劳无功。
2019-01-21 14:30
回复
前端看起来就一个“接口”,看起来可不是简单多了,哈哈。在他们眼里,下一步让前端直接传递sql更省事。
2019-01-21 14:37
回复
用户头像
"他们的遗留 REST API 数量暴增,变得十分复杂;"
"GET https://inbox.indeed.com/api/getConversationCount"

嗯,所以作者把RPC当做REST API了, 所以“他们的REST API“暴增了。
2019-01-21 14:27
回复
用户头像
文中举了个例子,"在使用 GraphQL 时,上面的这些请求可以被包含在单个查询和单个请求中。query HomePage",RestAPI也可以做到,“GET /xxx/api/homePage"。 所以GraphQL的优势在哪里?
2019-01-21 10:14
回复
RestAPI不仅指service 路径的pattern, 每个api都应该是独立而稳定的。 homePage是一个不稳定的resource, 会根据前端业务的变化而变化, 不应该在后端提供一个这样的api。
2019-01-22 11:42
回复
没有更多了
发现更多内容

MySQL主从配置+ThinkPHP5分布式数据库

北桥苏

MySQL 分布式 thinkphp

Prompt learning 教学[进阶篇]:简介Prompt框架并给出自然语言处理技术:Few-Shot Prompting、Self-Consistency等;项目实战搭建知识库内容机器人

汀丶人工智能

人工智能 深度学习 ChatGPT prompt learning

智慧园区移动应用发展面临瓶颈,如何解决?

没有用户名丶

软件测试 | admin后台管理

测吧(北京)科技有限公司

测试

2023 开源之夏来啦!报名 MegEngine 项目,赢取万元奖金!

MegEngineBot

深度学习 开源之夏 MegEngine 学生比赛 奖金

Django查询、删除、更新数据

测吧(北京)科技有限公司

测试

网页版超强ChatGPT插件应用ZipZap来了,每日免费10K Token足够使用

Ricky

openai ChatGPT GPT-4

智能化生产应用搭建的实战案例

华为云开发者联盟

后端 物联网 华为云 华为云开发者联盟 企业号 5 月 PK 榜

Java面试通关:阿里内部实战模拟面试精讲题库,竟被上传GitHub!

Java你猿哥

Java 数据库 微服务 Spring Boot java面试

一种KV存储的GC优化实践

vivo互联网技术

KV存储 GC设计

京东短网址高可用提升最佳实践 | 京东云技术团队

京东科技开发者

最佳实践 高可用设计 企业号 5 月 PK 榜 短网址

数据治理三大模式详解,治理新范式释放数据潜能

袋鼠云数栈

数据治理

开发中关于HTTPS不得不规避的2个大坑

北桥苏

https TCP/IP ssl

【开发点滴】在vscode的python调试中,如何让调试程序以sudo方式运行?

无人之路

Python vscode ChatGPT

面试官:说说 WebSocket 和 Socket 及 Http 的区别?

Java你猿哥

Java TCP ssm HTTP webSock

Fabarta 参加 2023 数云原力大会,与各方共同发布《2023 数据资产盘点实践白皮书》

Fabarta

数据要素 数据资产管理 数据资产化 数据要素流通

Django基本数据访问

测吧(北京)科技有限公司

测试

硬核!阿里最新出品架构核心场景实战手册,解决99%的架构问题

Java你猿哥

详解集群级备份恢复:物理细粒度备份恢复

华为云开发者联盟

数据库 后端 华为云 华为云开发者联盟 企业号 5 月 PK 榜

以数据思维和技能提升数据应用测试实践 | 京东云技术团队

京东科技开发者

测试 测试覆盖率 数据思维 应用安全测试 企业号 5 月 PK 榜

GreatSQL社区月报 | 2023.04

GreatSQL

greatsql社区

穿越数据智能“海峡”,企业更需要什么样的数智底座?

用友BIP

BATM面试Java岗:精选200+面试题及答案、6大重点规划和经验总结

Java你猿哥

Java MySQL JVM 多线程 java面试

中国20强(上市)游戏公司2022年财报分析:营收结构优化,市场竞争进入白热化

易观分析

公司 游戏

Alibaba技术专家倾心五年打造!Java工程师成神之路(基础篇)

做梦都在改BUG

Java

数据库外网ip binlog主从配置

Java你猿哥

Java MySQL 后端 ssm

微信小程序二维码文件流上传到OSS解决方法

北桥苏

php OSS thinkphp

【5.05-5.12】写作社区优秀技术博文一览

InfoQ写作社区官方

热门活动 优质创作周报

软件测试是青春饭吗?

测吧(北京)科技有限公司

测试

如何解决由于TLS版本不兼容导致的问题

JefferLiu

Java TLS

夺冠秘诀?华为软件精英挑战赛两届冠军这样复盘比赛经验

华为云开发者联盟

云计算 华为云 华为云开发者联盟 企业号 5 月 PK 榜

为什么说GraphQL可以取代REST API?_大前端_Azat Mardan_InfoQ精选文章