- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
所以我有这个引用博客的用户模型
用户模型
const mongoose = require("mongoose");
const Schema = mongoose.Schema;
const bcrypt = require("bcryptjs");
const userSchema = new Schema(
{
email: {
type: String,
required: true,
index: {
unique: true
}
},
password: {
type: String,
required: true
},
name: {
type: String,
required: true
},
website: {
type: String
},
bio: {
type: String
},
blogs: [
{
type: Schema.Types.ObjectId,
ref: "Blog"
}
]
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
}
}
);
userSchema.pre("save", function(next) {
const user = this;
if (!user.isModified("password")) return next();
bcrypt.genSalt(10, function(err, salt) {
if (err) return next(err);
bcrypt.hash(user.password, salt, function(err, hash) {
if (err) return next(err);
user.password = hash;
next();
});
});
});
userSchema.methods.comparePassword = function(password, next) {
bcrypt.compare(password, this.password, function(err, isMatch) {
if (err) return next(err);
next(null, isMatch);
});
};
const User = mongoose.model("User", userSchema);
module.exports = User;
这是我的博客集合,其中引用了评论模型
const mongoose = require("mongoose");
const Schema = mongoose.Schema;
const blogSchema = new Schema(
{
title: {
type: String,
required: true
},
body: {
type: String,
required: true
},
author: {
type: Schema.Types.ObjectId,
ref: "User"
},
likesCount: {
type: Number
},
comments: [
{
type: Schema.Types.ObjectId,
ref: "Comment"
}
]
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
}
}
);
const Blog = mongoose.model("Blog", blogSchema);
module.exports = Blog;
这是我的评论模型,其中引用了用户模型
const mongoose = require("mongoose");
const Schema = mongoose.Schema;
const CommentSchema = new Schema(
{
body: {
type: String,
required: true
},
likesCount: {
type: Number
},
user: {
type: Schema.Types.ObjectId,
ref: "User"
}
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
}
}
);
const Comment = mongoose.model("Comment", CommentSchema);
module.exports = Comment;
我想要的是,如果我获取用户数据,我想获取博客以及评论数据 我有这个代码
exports.getCurrentUser = async (req, res) => {
const ObjectId = mongoose.Types.ObjectId;
const users = await User.findById({ _id: new ObjectId(req.user._id) })
.populate({
path: "blogs",
model: "Blog"
})
.exec();
console.log(users);
return res.status(200).json(users);
};
但它没有填充博客
如何实现这种嵌套引用获取?
最佳答案
我认为问题在于 model 必须是模型引用,而不是模型名称。所以它必须是 model: Blog
而不是 model: "Blog"
。
此外,我建议您重新设计架构,因为模型之间有很多引用。当您添加、插入或删除博客或评论时,您将必须进行 2 次 db 调用。
我将从用户架构中删除博客字段,并从博客架构中删除评论字段,并设置虚拟填充,如下所示:
用户架构:
const userSchema = new Schema(
{
email: {
type: String,
required: true,
index: {
unique: true
}
},
password: {
type: String,
required: true
},
name: {
type: String,
required: true
},
website: {
type: String
},
bio: {
type: String
}
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
},
toJSON: {
virtuals: true
}
}
);
userSchema.virtual("blogs", {
ref: "Blog",
foreignField: "author",
localField: "_id"
});
博客架构:
const blogSchema = new Schema(
{
title: {
type: String,
required: true
},
body: {
type: String,
required: true
},
author: {
type: Schema.Types.ObjectId,
ref: "User"
},
likesCount: {
type: Number
}
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
},
toJSON: { virtuals: true }
}
);
blogSchema.virtual("comments", {
ref: "Comment",
foreignField: "blog",
localField: "_id"
});
请注意,我向这两个架构添加了 toJSON: { virtuals: true }
选项。
现在您可以使用以下查询获取带有评论的用户博客:
const user = await User.findById(req.user._id)
.populate({
path: "blogs",
populate: "comments"
})
.select("-password")
.exec();
测试:
假设您有这些示例文档。
db={
"users": [
{
"_id": "5e53b1726f41c765fc4def9c",
"email": "user1@gmail.com",
"password": "$2a$10$.heEhkN2BhxZiw8upgjGQe.r3Gt78JVfuAqLqf6lHwireaKJSrrTO",
"name": "User1"
},
{
"_id": "5e53b1906f41c765fc4def9d",
"email": "user2@gmail.com",
"password": "$2a$10$tEaXpoeH5iXVzqmzozAFOOu.Nxb32Ssy1XS5CAqad7qqanHQkrqjK",
"name": "User2"
},
{
"_id": "5e53b1996f41c765fc4def9e",
"email": "user3@gmail.com",
"password": "$2a$10$4s34RLnSd75WeG8K.gzxxeixkruplzW0vpb7PJR/aL1d3Ia31wj.W",
"name": "User3"
}
],
"blogs": [
{
"_id": "5e53b26c6f41c765fc4def9f",
"title": "Blog1 Title",
"body": "Blog1 Body",
"author": "5e53b1726f41c765fc4def9c"
},
{
"_id": "5e53b2896f41c765fc4defa1",
"title": "Blog2 Title",
"body": "Blog2 Body",
"author": "5e53b1726f41c765fc4def9c"
}
],
"comments": [
{
"_id": "5e53b2f86f41c765fc4defa3",
"body": "Comment1 (user2 on user1's blog1)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b3246f41c765fc4defa4",
"body": "Comment2 (user3 on user1's blog1)",
"user": "5e53b1996f41c765fc4def9e",
"blog": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b34c6f41c765fc4defa5",
"body": "Comment3 (user2 on user1's blog2)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b2896f41c765fc4defa1"
}
]
}
对于_id:“5e53b1726f41c765fc4def9c”的用户,结果将如下所示
:
{
"_id": "5e53b1726f41c765fc4def9c",
"email": "user1@gmail.com",
"name": "User1",
"created_at": "2020-02-24T11:20:18.343Z",
"updated_at": "2020-02-24T11:20:18.343Z",
"__v": 0,
"blogs": [
{
"_id": "5e53b26c6f41c765fc4def9f",
"title": "Blog1 Title",
"body": "Blog1 Body",
"author": "5e53b1726f41c765fc4def9c",
"created_at": "2020-02-24T11:24:28.895Z",
"updated_at": "2020-02-24T11:24:28.895Z",
"__v": 0,
"comments": [
{
"_id": "5e53b2f86f41c765fc4defa3",
"body": "Comment1 (user2 on user1's blog1)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b26c6f41c765fc4def9f",
"created_at": "2020-02-24T11:26:48.506Z",
"updated_at": "2020-02-24T11:26:48.506Z",
"__v": 0
},
{
"_id": "5e53b3246f41c765fc4defa4",
"body": "Comment2 (user3 on user1's blog1)",
"user": "5e53b1996f41c765fc4def9e",
"blog": "5e53b26c6f41c765fc4def9f",
"created_at": "2020-02-24T11:27:32.305Z",
"updated_at": "2020-02-24T11:27:32.305Z",
"__v": 0
}
],
"id": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b2896f41c765fc4defa1",
"title": "Blog2 Title",
"body": "Blog2 Body",
"author": "5e53b1726f41c765fc4def9c",
"created_at": "2020-02-24T11:24:57.078Z",
"updated_at": "2020-02-24T11:24:57.078Z",
"__v": 0,
"comments": [
{
"_id": "5e53b34c6f41c765fc4defa5",
"body": "Comment3 (user2 on user1's blog2)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b2896f41c765fc4defa1",
"created_at": "2020-02-24T11:28:12.551Z",
"updated_at": "2020-02-24T11:28:12.551Z",
"__v": 0
}
],
"id": "5e53b2896f41c765fc4defa1"
}
],
"id": "5e53b1726f41c765fc4def9c"
}
另一个选择是使用 MongoDB 聚合框架。使用此选项,您可以删除我为虚拟填充添加的选项。
const users = await User.aggregate([
{
$match: {
_id: req.user._id
}
},
{
$project: {
password: 0
}
},
{
$lookup: {
from: "blogs",
let: {
userId: "$_id"
},
pipeline: [
{
$match: {
$expr: {
$eq: ["$$userId", "$author"]
}
}
},
{
$lookup: {
from: "comments",
let: {
blogId: "$_id"
},
pipeline: [
{
$match: {
$expr: {
$eq: ["$blog", "$$blogId"]
}
}
}
],
as: "comments"
}
}
],
as: "blogs"
}
}
]);
users[0]
将为您提供与选项 1 相同的结果。
关于node.js - Mongoose 填充不填充嵌套数组数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60373751/
当我保存4条记录时,我需要将它们一一保存,所以我的代码是 a.save(function(){ b.save(function(){ c.save(function(){ d.save(f
Mongoose在版本4.2.7中创建了一个新的单个子文档功能(documentation和feature request),允许使用单个嵌入式子文档架构,其行为方式与一对多子文档的行为相同。 在父级
Mongoose 版本 >= 4.0 有一个时间戳选项,当 timestamps 设置为 时,该选项会为架构创建 updatedAt 和 createdAt 字段正确。 http://mongoose
我注意到某些图书馆喜欢 mock 鹅 ( https://github.com/mccormicka/Mockgoose/blob/master/test/index.spec.js ) 使用 req
我正在与 Mongoose 合作。我见过很多开发者发出以下命令: mongoose.Promise = global.Promise; 然后我很好奇 mongoose.Promise 的原始值是什么。
当我运行与数据库大量连接和断开连接的测试时,我收到以下警告。 (node) warning: possible EventEmitter memory leak detected. 11 connec
我可以在 Mongoose 的子文档数组中填充动态引用(使用“refPath”)虚拟字段吗? 数据结构如下 Group - Members -> User 代码:模型/模式 let MemberSc
我正在我的应用程序中做一些测试,看看内存缓存是否真的在工作。但是,由于 memory-cache 显然没有公开“命中”事件,我无法判断是否真的从缓存中获取数据。所以我试着看看当应用程序实际从数据库中获
我是 nestjs 的新手。我使用 @nestjs/mongoose,我需要在我的类模式中引用嵌套对象中的几个字段,但我不知道该怎么做。 dietDays 对象必须包含一个日期字段和包含对 Meal
我是 mongodb 的新手,我有一个这样的数据模型 { last_updated: Date.now(), csgo_items:[ {name: 'name', p
这是我的方案: var documentSchema = mongoose.Schema({ 'facts': [{ 'type': { type: String, requi
我想删除多个 _ids = ['123', '234', '345']; _ids.forEach(_id => { await model.deleteOne({ _id }); }); 有没有
我有一个像这样的 Mongoose 模式: var Address = { doorNo:String, city:String, state:String, coun
我的文档包含一个名为 clients 的字段那应该包含一组客户端ID。 { "first_name":"Nick", "last_name":"Parsons", "email":"nic
我遇到了以下我无法理解的代码行,尽管有很多教程提供了与 populate 的示例相关的信息。但没有一个能解释它究竟意味着什么。这是一个例子 var mongoose = require('mongoo
我有一个具有多个唯一性的架构,如下所示: var userSchema = new mongoose.Schema({ user: { type: String, unique:
我有一个 Mongoose 模式,其中有 4 个子模式。我一直在关注这里的文档https://github.com/LearnBoost/mongoose关于嵌入文档 var scenarios =
我希望每个字符串属性都默认设置为 true。有办法吗? ?? mongoose.Schema.String -> default { trim: true } var schema = new Sch
我有这个代码 var ClientSchema = new Schema({ name: {type: String, required: true, trim: true} }); var Cl
许多教程告诉您在您的 userSchema 页面中使用 bycrypt。保存新用户后,它会附带加密密码。伟大的。然而,我想,当我用某些东西编辑用户时,它也会重新哈希密码,导致无法登录。你能给我一个解决
我是一名优秀的程序员,十分优秀!