当前位置: 首页 > news >正文

[C#]项目中如何用 GraphQL 代替传统 WebAPI服务

在现代应用程序开发中,传统的 WebAPI 通常使用 RESTful 设计风格,然而近年来 GraphQL 作为一种新的 API 查询语言逐渐获得广泛应用。GraphQL 允许客户端精确地查询所需的数据,减少了过度请求和不足请求的问题。本文将详细讨论在项目中用 GraphQL 代替 WebAPI 的场景,包括它的优势、如何迁移、技术实现,以及潜在的挑战。

1. WebAPI 和 GraphQL 的对比

1.1 WebAPI(RESTful)的特点
  • 固定的端点:每个资源都有一个固定的 URL 端点,如 /users, /orders
  • 使用 HTTP 动词:通过 GETPOSTPUTDELETE 等 HTTP 动词执行不同的操作。
  • 过度请求/不足请求问题:客户端可能会获取过多不需要的数据,或不得不通过多次请求来获取所需信息。
  • 无状态:服务器不存储客户端状态,每个请求都是独立的。
1.2 GraphQL 的特点
  • 单一端点:GraphQL 使用单一的 /graphql 端点,客户端通过查询语言请求不同的数据。
  • 灵活的数据查询:客户端可以明确指定所需的数据,避免过度或不足请求。
  • 类型化查询:GraphQL 提供了一个强类型的模式(schema),使得查询结果结构化和可预测。
  • 支持实时更新:通过 GraphQL 的 Subscriptions,客户端可以订阅实时数据变化。

2. 为什么选择 GraphQL 替代 WebAPI?

2.1 减少数据传输

GraphQL 允许客户端精确指定需要的字段,而不像 REST API 那样会返回整个资源。这在数据传输量大、客户端性能受限的移动应用中尤为重要。

2.2 减少客户端请求

在传统 REST API 中,可能需要多个请求来获取复杂对象的嵌套数据。而 GraphQL 能够通过一次查询请求返回嵌套的相关数据,大大减少了请求次数。

2.3 自定义请求

GraphQL 的查询完全由客户端控制,允许根据需求动态获取不同的数据集。对于复杂的前端应用,GraphQL 提供了高度的灵活性和自定义查询能力。

2.4 强类型模式

GraphQL 的类型化查询模式允许开发人员清晰地定义数据结构,并提供自描述的 API,使得前后端协作更加顺畅。开发者可以在开发时使用工具如 GraphQL Playground 检查模式和测试查询。

2.5 支持实时数据

GraphQL 的 Subscriptions 机制允许客户端订阅服务器的实时事件更新,使其非常适合构建需要动态更新数据的应用,如社交媒体、实时通知系统等。

3. 如何将项目从 WebAPI 迁移到 GraphQL?

3.1 定义 GraphQL Schema

GraphQL 的核心是模式(schema),它定义了可查询的数据结构以及查询的类型。模式通常包含三种操作类型:

  • Query:用于读取数据。
  • Mutation:用于修改数据(如创建、更新、删除操作)。
  • Subscription:用于订阅实时更新。
type Query {users: [User]user(id: ID!): User
}type Mutation {createUser(name: String!, age: Int!): User
}type Subscription {userAdded: User
}type User {id: ID!name: String!age: Int!
}
3.2 数据解析器(Resolvers)

解析器是 GraphQL 的核心,它决定了如何从数据库或其他服务中获取数据。每个模式字段对应一个解析器函数。

public class Query
{public List<User> GetUsers([Service] IUserService userService) => userService.GetAllUsers();public User GetUser([Service] IUserService userService, int id) => userService.GetUserById(id);
}
3.3 配置 GraphQL 服务器

在项目中,可以使用类似 HotChocolate 或 GraphQL.NET 等库来设置 GraphQL 服务。以下是一个使用 HotChocolate 的 .NET MAUI 应用配置示例:

public void ConfigureServices(IServiceCollection services)
{services.AddGraphQLServer().AddQueryType<Query>().AddMutationType<Mutation>().AddSubscriptionType<Subscription>();
}
3.4 替换 RESTful API

一旦定义了模式并实现了解析器,就可以将现有的 REST API 替换为 GraphQL 查询。例如,将 /api/users 替换为 GraphQL 查询:

{users {idnameage}
}
3.5 数据实时更新

通过 GraphQL 的 Subscriptions 实现实时功能。当有新用户添加时,客户端可以接收到推送数据:

subscription {userAdded {idnameage}
}

4. GraphQL 的优势与挑战

4.1 优势
  • 高效的数据请求:客户端仅请求所需数据,避免了带宽浪费。
  • 简化 API 版本控制:由于查询是客户端控制的,服务端无需频繁进行 API 版本更新。
  • 开发体验更好:GraphQL 提供的强类型支持和工具链使得开发调试更加便捷。
4.2 挑战
  • 学习曲线:GraphQL 相对于 REST 来说具有一定的学习曲线,特别是对于模式定义和解析器的理解。
  • 缓存机制较复杂:相比 REST,GraphQL 缓存机制较为复杂,需要专门设计和实现。
  • 可能造成过度查询:由于客户端可以查询任意深度的数据嵌套,可能导致性能问题。可以通过设置查询深度限制等策略进行控制。

5. 实战案例:在项目中实现 GraphQL API

5.1 创建用户服务

假设我们有一个基于用户管理的系统,首先定义一个用户服务接口:

public interface IUserService
{List<User> GetAllUsers();User GetUserById(int id);User CreateUser(string name, int age);
}

实现该服务:

public class UserService : IUserService
{private readonly List<User> _users = new();public List<User> GetAllUsers() => _users;public User GetUserById(int id) => _users.FirstOrDefault(u => u.Id == id);public User CreateUser(string name, int age){var user = new User { Id = _users.Count + 1, Name = name, Age = age };_users.Add(user);return user;}
}
5.2 实现 GraphQL 查询和变更

配置好查询和变更解析器后,可以通过 GraphQL API 操作用户数据。

mutation {createUser(name: "Alice", age: 30) {idname}
}{users {idnameage}
}

6. GraphQL 工具链与优化

6.1 GraphQL Playground

GraphQL Playground 是一个用于测试和调试 GraphQL 查询的强大工具。通过它,开发者可以快速验证查询并查看结果。

6.2 性能优化
  • 查询深度限制:限制客户端可以查询的嵌套深度,避免过度查询。
  • 数据批处理:使用数据加载器(DataLoader)来减少 N+1 查询问题,提高数据库查询效率。

7. 总结

在项目中用 GraphQL 代替 WebAPI,可以极大提升前后端交互的灵活性、减少不必要的数据传输并支持实时数据更新。然而,在引入 GraphQL 时,也需要考虑学习成本、缓存设计和性能优化等挑战。通过合理的模式设计和解析器实现,GraphQL 可以成为现代应用开发中的高效解决方案。

友情链接 graphql

相关文章:

[C#]项目中如何用 GraphQL 代替传统 WebAPI服务

在现代应用程序开发中&#xff0c;传统的 WebAPI 通常使用 RESTful 设计风格&#xff0c;然而近年来 GraphQL 作为一种新的 API 查询语言逐渐获得广泛应用。GraphQL 允许客户端精确地查询所需的数据&#xff0c;减少了过度请求和不足请求的问题。本文将详细讨论在项目中用 Grap…...

对后端返回的日期属性进行格式化(扩展 Spring MVC 的消息转换器)

格式化之前 格式化之后&#xff1a; 解决方式 方式一 在属性中加上注解&#xff0c;对日期进行格式化 JsonFormat(pattern "yyyy-MM-dd HH:mm:ss")private LocalDateTime createTime;//JsonFormat(pattern &quo…...

踩坑记录-用python解析php Laravel8生成的jwt token一直提示 Invalid audience

import jwtdef token_required(token):with open(storage/oauth-public.key, r) as f:public_key f.read()try:# 尝试使用当前算法解码 token&#xff0c;同时指定受众decoded jwt.decode(token, public_key, algorithms[RS256], options{"verify_aud": False})# p…...

使用IOT-Tree Server制作一个边缘计算设备(Arm Linux)

最近实现了一个小项目&#xff0c;现场有多个不同厂家的设备&#xff0c;用户需要对此进行简单的整合&#xff0c;并实现一些联动控制。 我使用了IOT-Tree Server这个软件轻松实现了&#xff0c;不外乎有如下过程&#xff1a; 1&#xff09;使用Modbus协议对接现有设备&#…...

(JAVA)B树和B+树的实现原理阐述

1. B 树 2-3树中&#xff0c;一个节点最多能有两个key&#xff0c;它的实现红黑树中适用对链接染色的方式去表达这两个key。下面将学习另一种树形结构B树&#xff0c;这种数据结构中&#xff0c;一个节点允许多余两个key的存在。 B树是一种树状数据结构&#xff0c;它能够存储…...

JC系列CAN通信说明

目录 一、CAN协议二、指令格式三、通信接线3.1、一对一通信3.2、组网通信 四、寄存器定义五、指令说明4、读取电源电压5、读取母线电流6、读取实时速度8、读取实时位置10、读取驱动器温度11、读取电机温度12、读取错误信息32、设定电流33、设定速度35、设定绝对位置37、设定相对…...

Ubuntu22——安装并配置局域网文件共享系统Samba

我们将共享目录设置为 /home/takway/share。以下是基于这个新目录的详细步骤&#xff1a; 在Ubuntu上安装并配置Samba 更新系统包列表 打开终端&#xff0c;执行以下命令来确保你的包列表是最新的&#xff1a; sudo apt update安装Samba 安装Samba及其相关工具&#xff1a; sud…...

HTML CSS 基础

HTML & CSS 基础 HTML一、HTML简介1、网页1.1 什么是网页1.2 什么是HTML1.3 网页的形成1.4总结 2、web标准2.1 为什么需要web标准2.2 Web 标准的构成 二、HTML 标签1、HTML 语法规范1.1基本语法概述1.2 标签关系 2、 HTML 基本结构标签2.1 第一个 HTML 网页2.2 基本结构标签…...

Nginx 使用 GeoIP 模块阻止特定国家 IP 地址的最佳实践

一、概述 为什么要阻止特定国家的 IP 地址&#xff1f; 在全球化的互联网上&#xff0c;网站和服务器可能会面对来自不同国家和地区的用户流量。虽然大多数情况下&#xff0c;我们希望网站能为全球用户提供服务&#xff0c;但在某些特定场景下&#xff0c;阻止来自特定国家的…...

vue3 + vite + cesium项目

GitHub - tingyuxuan2302/cesium-vue3-vite: 项目基于 vue3 vite cesium&#xff0c;已实现常见三维动画场&#xff0c;欢迎有兴趣的同学加入共建&#xff0c;官网服务器相对拉胯&#xff0c;请耐心等候...https://github.com/tingyuxuan2302/cesium-vue3-vite/tree/github...

DR模式 LVS负载均衡群集

DR模式 LVS负载均衡群集 部署共享存储关闭防火墙和核心防护下载&#xff0c;开启nfs服务创建共享文件夹和测试用的静态网页文件编辑nfs配置文件发布共享查看共享 配置 tomcat 服务器关闭防火墙和核心防护安装tomcat配置 tomcat 多实例 配置 nginx 服务器关闭防火墙和核心防护配…...

mysql复制表结构和数据

1.实例 #复制一张和test 一摸一样的表结构 CREATE TABLE test_one like test#往复制的表结构中复制数据 INSERT INTO test_one SELECT * FROM test#两者一起使用相当于 cv大法2.总结 完全实现了表结构和数据的复制&#xff0c;但是两条sql 得分两步执行 2.1 复制表结构 #复制…...

MFC扩展库BCGControlBar Pro v35.1新版亮点:改进网格控件性能

BCGControlBar库拥有500多个经过全面设计、测试和充分记录的MFC扩展类。 我们的组件可以轻松地集成到您的应用程序中&#xff0c;并为您节省数百个开发和调试时间。 BCGControlBar专业版 v35.1已全新发布了&#xff0c;这个版本改进网格控件的性能、增强工具栏编辑器功能等。 …...

Python列表操作详解

1 列表的基本概念 在Python中&#xff0c;列表是一种非常常用的数据结构&#xff0c;它可以存储任意类型的元素&#xff0c;并且支持多种操作。下面将详细介绍Python列表的各种操作。 2列表的操作方法 2.1创建列表 Python可以直接使用方括号[]来创建一个空列表。 示例&am…...

畅捷通T+对接聚水潭成功实施案例

在当今竞争激烈的商业环境中&#xff0c;企业数字化转型已成为提升竞争力的关键。广东某实业有限公司的数字化规划&#xff0c;目前财务系统使用的畅捷通T&#xff0c;电商系统使用的聚水潭。目前两个系统数据割裂导致各个部门的协同效率低下。通过借助轻易云数据集成平台&…...

leetcode-312. 戳气球

题目描述 有 n 个气球&#xff0c;编号为0 到 n - 1&#xff0c;每个气球上都标有一个数字&#xff0c;这些数字存在数组 nums 中。 现在要求你戳破所有的气球。戳破第 i 个气球&#xff0c;你可以获得 nums[i - 1] * nums[i] * nums[i 1] 枚硬币。 这里的 i - 1 和 i 1 代…...

程序设计基础I-实验7 函数(编程题)

7-1 sdut- C语言实验—计算表达式 计算下列表达式值&#xff1a; 输入格式: 输入x和n的值&#xff0c;其中x为非负实数&#xff0c;n为正整数。 输出格式: 输出f(x,n)&#xff0c;保留2位小数。 输入样例: 3 2输出样例: 在这里给出相应的输出。例如&#xff1a; 2.00 …...

使用3080ti配置安装blip2

使用3080ti运行blip2的案例 本机环境&#xff08;大家主要看GPU&#xff0c;ubuntu版本和cuda版本即可&#xff09;&#xff1a;安装流程我最后安装的所有包的信息&#xff08;python 3.9 &#xff09;以供参考&#xff08;environment.yml&#xff09;&#xff1a; 本机环境&a…...

vue3组件通信之defineEmits

一、defineEmits是什么&#xff1f; defineEmits 是vue3提供的方法&#xff0c;又称为自定义事件&#xff0c;不需要引入可以直接使用&#xff0c;用于子组件与父组件通信。 二、使用样例 1.父组件代码 代码如下&#xff08;示例&#xff09;&#xff1a; <template>…...

rust gio-rs 挂载 samba 磁盘

linux 使用的 gio 管理工具 这个工具如下 这是 gio 的rust版本 https://crates.io/crates/gio 可以用 rust 语言实现下面所有操作 gio mout 挂载 samba 如下 //https://valadoc.org/gio-2.0/GLib.MountOperation.html pub async fn gio_mount(uri路径:&str, 用户名:Opti…...

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…...

国防科技大学计算机基础课程笔记02信息编码

1.机内码和国标码 国标码就是我们非常熟悉的这个GB2312,但是因为都是16进制&#xff0c;因此这个了16进制的数据既可以翻译成为这个机器码&#xff0c;也可以翻译成为这个国标码&#xff0c;所以这个时候很容易会出现这个歧义的情况&#xff1b; 因此&#xff0c;我们的这个国…...

C++实现分布式网络通信框架RPC(3)--rpc调用端

目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中&#xff0c;我们已经大致实现了rpc服务端的各项功能代…...

高等数学(下)题型笔记(八)空间解析几何与向量代数

目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...

Axios请求超时重发机制

Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式&#xff1a; 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

[Java恶补day16] 238.除自身以外数组的乘积

给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O(n) 时间复杂度…...

【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习

禁止商业或二改转载&#xff0c;仅供自学使用&#xff0c;侵权必究&#xff0c;如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...

学习一下用鸿蒙​​DevEco Studio HarmonyOS5实现百度地图

在鸿蒙&#xff08;HarmonyOS5&#xff09;中集成百度地图&#xff0c;可以通过以下步骤和技术方案实现。结合鸿蒙的分布式能力和百度地图的API&#xff0c;可以构建跨设备的定位、导航和地图展示功能。 ​​1. 鸿蒙环境准备​​ ​​开发工具​​&#xff1a;下载安装 ​​De…...

在鸿蒙HarmonyOS 5中使用DevEco Studio实现指南针功能

指南针功能是许多位置服务应用的基础功能之一。下面我将详细介绍如何在HarmonyOS 5中使用DevEco Studio实现指南针功能。 1. 开发环境准备 确保已安装DevEco Studio 3.1或更高版本确保项目使用的是HarmonyOS 5.0 SDK在项目的module.json5中配置必要的权限 2. 权限配置 在mo…...

Vue 3 + WebSocket 实战:公司通知实时推送功能详解

&#x1f4e2; Vue 3 WebSocket 实战&#xff1a;公司通知实时推送功能详解 &#x1f4cc; 收藏 点赞 关注&#xff0c;项目中要用到推送功能时就不怕找不到了&#xff01; 实时通知是企业系统中常见的功能&#xff0c;比如&#xff1a;管理员发布通知后&#xff0c;所有用户…...