草庐IT

node.js - 在 MongoDB 中定义 3 个具有不同关系的模式(多对多,一对多......)

coder 2023-11-06 原文

我的应用程序中有 3 个不同的模式:

userSchema, questionSchema, listingSchema

三者的关系如下:

每个列表都有许多与之相关的问题(每个列表都有相同的问题)。
每个用户可以回答多个列表中的许多问题。
许多列表中的许多用户回答了每个问题

我正试图全神贯注地定义这些模式之间的正确关系(主要是因为诸如“_id = 100 的用户回答了 _id = 5 的问题之类的问题> 在带有 _id = 3 的列表中,我如何以最有效的方式更新所有这些?)。

到目前为止我已经定义了:

问题架构:

var questionSchema = new Schema({
    description:  String
});

用户架构:

var userSchema = new Schema({

    local            : {
        email        : String,
        password     : String,
        name         : String
    },
    ApartmentsAndQuestions: [{
        apartmentID: String,
        questionID: [String] /* one apartment -> multiple questions */
    }]
});

和listingSchema:

var listingSchema = new Schema({
     street          : String,
    buildingNumber  : Number,
    apartmentNumber : Number,
    type            : String,
    floor           : Number,
    outOfFloors     : Number,
    numberOfRooms   : Number,
    size            : Number,
    renovated       : Boolean,
    elevator        : Boolean,
    airConditioning : Boolean,
    balcony         : Boolean,
    price           : Number,
    description     : String,
    flagCount       : Number,
    pictures        : [imageSchema]
    owner           : [userSchema]

    UsersAndQuestions: [{
            userID: String,
            questionID: [String] /* one user -> multiple questions asked possible */
    }]
});

问题:如何在我的NoSQL数据库中做好?我的定义有意义吗?有没有更好的方法来描述这些模式之间的关系?

我们将非常感谢任何帮助!

最佳答案

MongoDB 3.2+ 解决方案

在评论中添加提及,可以使用新的$lookup以避免嵌入大量数据。这就像一个 SQL LEFT JOIN:

让我们添加一些数据,匹配你的:

db.questionSchema.insert({ _id: 1, description: "My description 1" });
db.questionSchema.insert({ _id: 2, description: "My description 2" });
db.questionSchema.insert({ _id: 3, description: "My description 3" });
db.questionSchema.insert({ _id: 4, description: "My description 4" });

db.userSchema.insert({ _id: 1, email: "my@email1.com", ApartmentsAndQuestions: [] });
db.userSchema.insert({ _id: 2, email: "my@email2.com", ApartmentsAndQuestions: [] });

db.listingSchema.insert({ _id: "A", UsersAndQuestions: [] })
db.listingSchema.insert({ _id: "B", UsersAndQuestions: [] })

// Add some questions
db.userSchema.update({ _id: 1 }, { $addToSet: { ApartmentsAndQuestions: { apartment_id: 1, question_id: [1] } } })
db.userSchema.update({ _id: 1, "ApartmentsAndQuestions.apartment_id": 1 }, { $addToSet: { "ApartmentsAndQuestions.$.question_id": 3 } })

db.userSchema.update({ _id: 2 }, { $addToSet: { ApartmentsAndQuestions: { apartment_id: 2, question_id: [1,2] } } })
db.userSchema.update({ _id: 2, "ApartmentsAndQuestions.apartment_id": 2 }, { $addToSet: { "ApartmentsAndQuestions.$.question_id": 4 } })

db.listingSchema.update({ _id: "A" }, { $addToSet: { UsersAndQuestions: { user_id: 1, question_id: [1] } } })

通过常规查找,您会得到以下结果:

test> db.listingSchema.find()
{
  "_id": "B",
  "UsersAndQuestions": [ ]
}
{
  "_id": "A",
  "UsersAndQuestions": [
    {
      "user_id": 1,
      "question_id": [
        1
      ]
    }
  ]
}

然后,让我们 $lookup:

db.listingSchema.aggregate([
    {
        $unwind: "$UsersAndQuestions"
    }
    ,{
        $lookup:
        {
            from: "userSchema",
            localField: "UsersAndQuestions.user_id",
            foreignField: "_id",
            as: "fetched_user"
        }
    }
    ,{
        $unwind: "$UsersAndQuestions.question_id"
    }
    ,{
        $lookup:
        {
            from: "questionSchema",
            localField: "UsersAndQuestions.question_id",
            foreignField: "_id",
            as: "fetched_question"
        }
    }
])

你得到:

{
  "waitedMS": NumberLong("0"),
  "result": [
    {
      "_id": "A",
      "UsersAndQuestions": {
        "user_id": 1,
        "question_id": 1
      },
      "fetched_user": [
        {
          "_id": 1,
          "email": "my@email1.com",
          "ApartmentsAndQuestions": [
            {
              "apartment_id": 1,
              "question_id": [
                1,
                3
              ]
            }
          ]
        }
      ],
      "fetched_question": [
        {
          "_id": 1,
          "description": "My description 1"
        }
      ]
    }
  ],
  "ok": 1
}

然后,您也可以 $unwind ApartmentsAndQuestions.questions_id 和 $lookup 问题数据。这取决于你。

关于node.js - 在 MongoDB 中定义 3 个具有不同关系的模式(多对多,一对多......),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34043495/

有关node.js - 在 MongoDB 中定义 3 个具有不同关系的模式(多对多,一对多......)的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

  3. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  4. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  5. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  6. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  7. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  8. ruby - 如何在 Grape 中定义哈希数组? - 2

    我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>

  9. ruby - 获取模块中定义的所有常量的值 - 2

    我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c

  10. ruby - 这两个 Ruby 类初始化定义有什么区别? - 2

    我正在阅读一本关于Ruby的书,作者在编写类初始化定义时使用的形式与他在本书前几节中使用的形式略有不同。它看起来像这样:classTicketattr_accessor:venue,:datedefinitialize(venue,date)self.venue=venueself.date=dateendend在本书的前几节中,它的定义如下:classTicketattr_accessor:venue,:datedefinitialize(venue,date)@venue=venue@date=dateendend在第一个示例中使用setter方法与在第二个示例中使用实例变量之间是

随机推荐