GraphQL 教程(六)—— N+1问题和缓存等问题

栏目: 编程工具 · 发布时间: 5年前

内容简介:考虑这样一个常见应用场景,前端页面上需要展示一个文章列表,其中包括了文章的标题,并且会同时显示每篇文章的作者名。那么我们可能会按下面几种方案来设计我们的API。方案一: 对于Restful,设计下面两个接口,客户端总共需要请求1+N次接口,查询数据库1+N次。方案二: 对于Restful,为避免多次频繁请求获取文章作者信息的接口,可以采取下面两种方式,不过都得修改后端代码,总共需要请求1次接口,查询数据库1+N次。

考虑这样一个常见应用场景,前端页面上需要展示一个文章列表,其中包括了文章的标题,并且会同时显示每篇文章的作者名。那么我们可能会按下面几种方案来设计我们的API。

方案一: 对于Restful,设计下面两个接口,客户端总共需要请求1+N次接口,查询数据库1+N次。

/articles	# 文章列表接口
/articles/{article_id}/author	# 获取文章作者信息接口

方案二: 对于Restful,为避免多次频繁请求获取文章作者信息的接口,可以采取下面两种方式,不过都得修改后端代码,总共需要请求1次接口,查询数据库1+N次。

/articles?withauthor=true		# 在原接口的基础上添加一个参数表示是否同时获取文章的作者信息
/articlesWithAuthor					# 直接添加一个新的能够通过获取作者信息的文章列表接口

方案三:对于GraphQL,我们不需要做任何的修改,因为我们早已经定义好这样的Schema将article与author进行了关联:

# schemas/article.py
class ArticleSchema(SQLAlchemyObjectType):
    author = graphene.Field("schemas.AuthorSchema", description="文章作者信息")

    def resolve_author(self, info):
        # 下面这一行请自行在demo中去掉注释
        return AuthorManager.get_one(id=self.author_id)

    class Meta:
        model = ArticleModel
        description = "文章Schema"

默认是打开了 SQL 日志的,所以我们可以看到最终的结果仍然是1次请求,数据库查询却有1+N次:

SELECT * FROM `articles` LIMIT 0, 20;
SELECT * FROM `authors` WHERE `id` = 1;
SELECT * FROM `authors` WHERE `id` = 2;
SELECT * FROM `authors` WHERE `id` = 3;
...
SELECT * FROM `authors` WHERE `id` = N;

为了防止N+1问题,社区为 GraphQL 提供了一个解决方案: DataLoader 。其原理就是,在需要查询数据库的时候将查询进行延迟,等到拿到所有的查询需求之后再一次性查询出来。在 graphene 里面,批量查询可以这样写:

# managers/author.py
class AuthorsDataLoader(DataLoader):
    def batch_load_fn(self, ids):
        query = DBSession().query(AuthorModel).filter(AuthorModel.id.in_(ids))
        articles = dict([(article.id, article) for article in query.all()])
        return Promise.resolve([articles.get(id, None) for id in ids])

最终,仅需要两次数据库查询就完成了两个批量查询,即:

SELECT * FROM `articles` LIMIT 0, 20;
SELECT * FROM `authors` WHERE `id` IN (1, 2, 3, ..., N);

至此,我们通过批量查询的方式完成了减少冗余查询的功能。但是性能问题依然存在。

缓存问题

仔细看上面的两条SQL,第一条查询由于是Graphene框架自己帮我完成了解析然后进行查询,所以其实并不是 SELECT * ,但是多级查询的时候,后面的查询逻辑是我自己写的,为了方便我就写的 SELECT * ,大家都应该知道这样子查询数据库会有很大的性能隐患,访问量一旦大了数据库压力会加倍增长。不过好处是, DataLoader 本身具有缓存结果的功能,它缓存的是 SELECT * FROM authors WHERE id =N 的结果,而不是批量查询的结果,所以,下一次即使是不一样的 IN 列表,依然会有部分能够使用缓存。 DataLoader 默认本身是开启缓存的,如果想自己用 Redis 等来实现对象的缓存,可以在 DataLoader 初始化的时候将 cache 设置为 False :

# schemas/.__init__.py
def get_dataloaders():
    return {
        "ArticlesDataLoader": ArticlesDataLoader(cache=False),
        "AuthorsDataLoader": AuthorsDataLoader(cache=False),
        "CommentsDataLoader": CommentsDataLoader(),
        "ArticleCommentsDataLoader": ArticleCommentsDataLoader(),
        "OrdinaryWritersDataLoader": OrdinaryWritersDataLoader(cache=True),
        "ProfessionalWritersDataLoader": ProfessionalWritersDataLoader(cache=False),
    }

验证问题/限流问题

由于只有一个查询入口,不能像Restful那样针对每个接口进行单独的验证或者限流。但是,正如我之前所说的,一个简单的方法就是强制让用户传入规范命名的查询语句,通过命名来进行后续的判断,就能方便地达到我们的要求。

分页查询问题

这个有很多种实现方式,大多数是用类似数据库 limit begin, end 的方式,但是为了兼容 Restful 的习惯,我使用的仍然是 Restful 风格的分页方式,返回结果也与 Restful 类似,可以参考代码:

# schemas/__init__.py
class PageInfoSchema(ObjectType):
    """
    专用于分页的schema
    """
    total = graphene.Int(description="总条数")
    current_page = graphene.Int(description="当前页码")
    per_page = graphene.Int(description="每页数量")
    total_pages = graphene.Int(description="总共页码数量")

    @staticmethod
    def paginate(total: int, current_page: int, per_page: int):
        """
        :param total: 总共条数
        :param current_page: 当前页码
        :param per_page: 每页数量
        :return:
        """
        return PageInfoSchema(
            total=int(total),
            current_page=int(current_page),
            per_page=int(per_page),
            total_pages=math.ceil(int(total) / int(per_page)),
        )

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

Effective Java 中文版

Effective Java 中文版

(美)Joshua Bloch / 潘爱民 / 机械工业出版社 / 2003-1 / 39.00元

本书介绍了在Java编程中57条极具实用价值的经验规则,这些经验规则涵盖了大多数开发人员每天所面临的问题的解决方案。通过对Java平台设计专家所使用的技术的全面描述,揭示了应该做什么,不应该做什么才能产生清晰、健壮的高效的代码。 本书中的每条规则都以简短、独立的小文章形式出现,这些小文章包含了详细而精确的建议,以及对语言中许多细微之处的深入分析,并通过例子代码加以进一步说明。贯穿全书的是通用......一起来看看 《Effective Java 中文版》 这本书的介绍吧!

URL 编码/解码
URL 编码/解码

URL 编码/解码

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具