mongodb - 蒙哥 : how to sort by external weight

标签 mongodb mongoid aggregation-framework

关注 this question @NeilLunn 已经优雅地回答了,这是我更详细的问题。

这是一组文档,有些有 user_id 有些没有。 user_id 代表创建文档的用户:

{ "user_id" : 11, "content" : "black", "date": somedate }
{ "user_id" : 6, "content" : "blue", "date": somedate }
{ "user_id" : 3, "content" : "red", "date": somedate }
{ "user_id" : 4, "content" : "black", "date": somedate }
{ "user_id" : 4, "content" : "blue", "date": somedate }
{ "user_id" : 90, "content" : "red", "date": somedate }
{ "user_id" : 7, "content" : "orange", "date": somedate }
{ "content" : "orange", "date": somedate }
{ "content" : "red", "date": somedate }
...
{ "user_id" : 4, "content" : "orange", "date": somedate }
{ "user_id" : 1, "content" : "orange", "date": somedate }
{ "content" : "red", "date": somedate }
{ "user_id" : 90, "content" : "purple", "date": somedate }

前端是拉页面,所以每个页面会有10个项目,我用limit和skip来做,效果很好。

如果我们有一个登录用户,我想根据他与之交互的用户,向当前登录的用户显示他可能首先发现更有趣的文档。

当前用户可能感兴趣的用户列表按分数排序,位于 mongo 之外。所以第一个元素是最重要的用户,我想首先显示他的文档,而列表中的最后一个用户是最不重要的。

列表是一个简单的数组,如下所示: [4,7,90,1]。

创建此用户分数的系统不在 mongo 中,但如果有帮助,我可以复制数据。我还可以更改数组以包含分数。

我想要完成的是:

从列表中获取按照 user_id 的重要性排序的文档,以便 user_id 4 的文档首先显示,user_id 7 的文档第二个显示,依此类推。当列表中没有用户时,我想显示其余的文档。像这样:

  1. user_d:4 的所有文档
  2. user_d:7 的所有文档
  3. user_d:90 的所有文档
  4. user_d:1 的所有文档
  5. 所有其余文件

我应该如何做到这一点?我是不是对 mongo 要求太多了?

最佳答案

给定数组 [4,7,90,1] 你想要的查询是这样的:

db.collection.aggregate([
   { "$project": {
       "user_id": 1,
       "content": 1,
       "date": 1,
       "weight": { "$or": [
           { "$eq": ["$user_id": 4] }, 
           { "$eq": ["$user_id": 7] }, 
           { "$eq": ["$user_id": 90] }, 
           { "$eq": ["$user_id": 1] }, 
       ]}
   }},
   { "$sort": { "weight": -1, "date": -1 } }
])

因此,对于包含在该 $or 条件中的每个项目,user_id 字段将根据提供的值进行测试,并且 $eqtruefalse 返回 10

您在代码中所做的是针对您构建 $or 的数组条件的数组中的每个项目。所以它只是为每个等于条件创建一个哈希结构,将它传递给一个数组并将其作为 $or 条件的数组值插入。

我可能应该把 $cond 运算符从前面的代码中去掉,这样这部分会更清楚。

以下是 Ruby Brain 的一些代码:

userList = [4, 7, 90, 1];

orCond = [];

userList.each do |userId|
  orCond.push({ '$eq' => [ 'user_id', userId ] })
end

pipeline = [
    { '$project' => {
        'user_id' => 1,
        'content' => 1,
        'date' => 1,
        'weight' => { '$or' => orCond }
    }},
    { '$sort' => { 'weight' => -1, 'date' => -1 } }
]

如果您想要单独的权重并且我们假设键值对,那么您需要使用 $cond 嵌套:

db.collection.aggregate([
   { "$project": {
       "user_id": 1,
       "content": 1,
       "date": 1,
       "weight": { "$cond": [
           { "$eq": ["$user_id": 4] },
           10,
           { "$cond": [ 
               { "$eq": ["$user_id": 7] },
               9,
               { "$cond": [
                   { "$eq": ["$user_id": 90] },
                   7,
                   { "$cond": [
                       { "$eq": ["$user_id": 1] },
                       8, 
                       0
                   ]}
               ]}
           ]}
       ]}
   }},
   { "$sort": { "weight": -1, "date": -1 } }
])

注意,它只是一个返回值,这些不需要按顺序排列。你可以想想它的产生。

要生成此结构,请参见此处:

https://stackoverflow.com/a/22213246/2313887

关于mongodb - 蒙哥 : how to sort by external weight,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22192098/

相关文章:

蒙戈数据库 4.2.8 : Unable to add session into the cache because the number of active sessions is too high

mongodb - 在 Mongoid 中选择 embeds_many 和references_many

mongodb - 通过 MongoDB 中的 collection.update 将所有数组值更新为Upper

mongodb - 合并 MongoDB 中的两个 $lookup 集合

javascript - 如何展平聚合输出的 JSON 结果

mongodb - MongoDB聚合管道中的平方根

java - 使用字段条件从 mongodb 集合中检索文档

node.js - 将 Mongoose 查询结果存储在另一个 Mongoose 查询中

ruby-on-rails - mongoid 查询集合,其中两列彼此相等

mongoid - 如何使用 Mongoid 重新索引集合?