我想从VideoSchema中按限制、偏移量和观众顺序获得类别

const CategoriesSchema = new Schema<ICategories>({
  name: String,
  image: String,
});

const VideoSchema = new Schema<IVideo>({
  videoSource: String,
  // from CategoriesSchema
  category: {
    name: String,
    image: String,
  },
  viewers: Number,
});

我要做什么才能得到我想要的:

const resolver = async ({ args: { limit, offset } }) => {
      // there are 60k+ categories so request will be long
      const categories = await CategoriesModel.find();
      for (let i = 0; i < categories.length; i++) {
        const streams = await VideosModel.find({ 'category.name': categories[i].name });
        categories[i] = {
          ...categories[i],
          // count viewers of current video
          viewers: streams.reduce((acc, curr) => acc + curr.viewers, 0),
        };
      }
      // sort by viewers, slice array by limit and offset
      return categories.sort(({ viewers: viewersA = 0 }, { viewers: viewersB = 0 }) => viewersB - viewersA).slice(offset, limit);
  },

我知道请求没有得到优化,但我不知道如何按限制和偏移量进行请求,并按当前的浏览人数对其进行排序.我知道在像mysql这样的关系数据库中,我可以使用left-join来实现这种情况,但我不知道如何使用mongoose来实现这种功能

清理:

enter image description here

有一个类别列表,每个类别都有来自VideoSchema的观众数量(观众字段).如果该类别没有视频,则应该有0个观众,或者如果我们有该类别的视频,则打印观众数量,从0到2^32

此外,我在VideoSchema中进行了更改,现在它引用了类别

const VideoSchema = new Schema<IVideo>({
  videoSource: String,
  // from CategoriesSchema
  category: {
    ref: 'Categories'
  },
  viewers: Number,
});

推荐答案

这是否回答了您的问题:

如果我理解正确的话,你想要每category个,从相关的videos中加上viewers的总和,排序并切片.

在mongoDB上,您可以轻松地在一个查询中完成这一切.由于您也希望获得具有0个查看器的类别,我们将从category个集合开始:

  1. $lookup以获得每个类别的所有videos个,并将$lookup个管道内每个类别的查看器数量相加,以最小化集合之间的数据转换.
  2. 格式化video号数据的响应和处理 case
  3. $sort乘以viewers$skip$limit-用于切片.将_id添加到$sort作为第二个选项,以便即使对于0个观众类别也能保持其排序.
db.category.aggregate([
  {
    $lookup: {
      from: "video",
      let: {category_id: "$_id"},
      pipeline: [
        {$match: {$expr: {$eq: ["$category", "$$category_id"]}}},
        {$group: {_id: 0, viewers: {$sum: "$viewers"}}}
      ],
      as: "viewers"
    }
  },
  {$set: {viewers: {$ifNull: [{$first: "$viewers.viewers" }, 0]}}},
  {$sort: {viewers: -1, _id: 1}},
  {$skip: offset},
  {$limit: limit}
])

看看它在playground example上是如何工作的

Javascript相关问答推荐

colored颜色 检测JS,平均图像 colored颜色 检测JS

从Node JS将对象数组中的数据插入Postgres表

为什么Mutations 观察器用微任务队列而不是macrotask队列处理?

在react JS中映射数组对象的嵌套数据

未定义引用错误:未定义&Quot;而不是&Quot;ReferenceError:在初始化&Quot;之前无法访问';a';

查询参数未在我的Next.js应用路由接口中定义

如何在.NET Core中将chtml文件链接到Java脚本文件?

将异步回调转换为异步生成器模式

SPAN不会在点击时关闭模式,尽管它们可以发送日志(log)等

JS Animate()方法未按预期工作

警告框不显示包含HTML输入字段的总和

使用RxJS from Event和@ViewChild vs KeyUp事件和RxJS主题更改输入字段值

无法设置RazorPay订阅API项目价格

使用API调用的VUE 3键盘输入同步问题

如何使用[ModelJSON,ArrayBuffer]调用tf.loadGraphModelSync

如何在Reaction中清除输入字段

Chart.js Hover线条在鼠标离开时不会消失

如何使用Cypress在IFRAME中打字

在一次操作中使用多个addToSet不适用于多个数组

无限循环,因为变量不断被重新声明为其原始值