在 Mongoose 中实现用户添加好友功能:最佳实践指南

在 mongoose 中实现用户添加好友功能:最佳实践指南

本文旨在指导开发者如何在 Mongoose 中实现用户添加好友的功能,重点讨论了如何处理好友请求、更新用户的好友列表,以及避免潜在的数据一致性问题。文章将探讨使用 FriendRequest 模型来管理好友关系,并分析直接在 User 模型中维护好友列表的优缺点,同时提供相应的代码示例和注意事项。

处理好友请求

首先,我们需要一个机制来处理好友请求。一个常见的做法是创建一个 FriendRequest 模型,用于存储好友请求的发送者、接收者和状态。

FriendRequest Schema:

const mongoose = require('mongoose');const friendRequestSchema = new mongoose.Schema({  sender: {    type: mongoose.Schema.Types.ObjectId,    ref: "User",    required: true,  },  receiver: {    type: mongoose.Schema.Types.ObjectId,    ref: "User",    required: true,  },  status: {    type: String,    enum: ["pending", "accepted", "rejected"],    default: "pending",  },  createdAt: {    type: Date,    default: Date.now,  },});const FriendRequest = mongoose.model('FriendRequest', friendRequestSchema);module.exports = FriendRequest;

发送好友请求的路由:

在发送好友请求的路由中,我们需要确保只有经过身份验证的用户才能发送请求。 建议使用中间件来验证用户身份,并从已验证的用户信息中获取发送者的 ID,而不是从前端传递。这样可以避免潜在的安全风险,例如用户篡改请求中的发送者 ID。

router.post("/requests", (req, res) => {  const { receiver } = req.body;  FriendRequest.create({    sender: req.user._id, // 从经过身份验证的用户获取    receiver,  })  .then(() => {    res.status(204).send();  })  .catch((error) => {    console.error(error); // 记录错误信息    res.status(500).send("Error sending friend request");  });});

处理好友请求的接受/拒绝

当用户接受或拒绝好友请求时,我们需要更新 FriendRequest 模型的 status 字段。

router.post("/requests/:id/accept", async (req, res) => {  try {    const request = await FriendRequest.findById(req.params.id);    if (!request) {      return res.status(404).send("Friend request not found");    }    request.status = "accepted";    await request.save();    // TODO: 更新 User 模型的好友列表 (见下文)    res.redirect("/requests");  } catch (error) {    console.error(error);    res.status(500).send("Error accepting friend request");  }});router.post("/requests/:id/reject", async (req, res) => {  try {    const request = await FriendRequest.findById(req.params.id);    if (!request) {      return res.status(404).send("Friend request not found");    }    request.status = "rejected";    await request.save();    res.redirect("/requests");  } catch (error) {    console.error(error);    res.status(500).send("Error rejecting friend request");  }});

更新 User 模型的好友列表

现在,让我们讨论如何更新 User 模型的好友列表。 有两种主要方法:

直接在 User 模型中维护好友列表: 在 User 模型中添加一个 friends 数组,存储好友的 ID。通过查询 FriendRequest 模型获取好友列表: 不在 User 模型中存储好友列表,而是通过查询 FriendRequest 模型,查找所有 status 为 “accepted” 的好友请求,来动态获取好友列表。

方法 1:直接在 User 模型中维护好友列表

// User Schemaconst userSchema = new mongoose.Schema({  firstname: String,  lastname: String,  username: {    type: String,    unique: true,    required: [true, "can't be blank"],    match: [/^[a-zA-Z0-9]+$/, "is invalid"],    index: true,  },  password: String,  createdAt: { type: Date, default: Date.now },  profilePictureURL: String,  friends: [    {      type: mongoose.Schema.Types.ObjectId,      ref: "User",    },  ],});const User = mongoose.model('User', userSchema);module.exports = User;// 在接受好友请求的路由中更新 User 模型router.post("/requests/:id/accept", async (req, res) => {  try {    const request = await FriendRequest.findById(req.params.id);    if (!request) {      return res.status(404).send("Friend request not found");    }    request.status = "accepted";    await request.save();    const senderId = request.sender;    const receiverId = request.receiver;    // 更新发送者的好友列表    const sender = await User.findById(senderId);    if (!sender) {      return res.status(404).send("Sender not found");    }    sender.friends.push(receiverId);    await sender.save();    // 更新接收者的好友列表    const receiver = await User.findById(receiverId);    if (!receiver) {      return res.status(404).send("Receiver not found");    }    receiver.friends.push(senderId);    await receiver.save();    res.redirect("/requests");  } catch (error) {    console.error(error);    res.status(500).send("Error accepting friend request");  }});

注意事项:

事务: 同时更新两个 User 模型的 friends 数组是一个潜在的数据一致性问题。 如果在更新其中一个模型时发生错误,可能会导致数据不一致。 为了解决这个问题,应该使用 Mongoose 的事务功能。错误处理: 在更新 User 模型之前,应该检查用户是否存在。

方法 2:通过查询 FriendRequest 模型获取好友列表

这种方法避免了在 User 模型中维护冗余数据。 要获取用户的好友列表,只需查询 FriendRequest 模型,查找所有 sender 或 receiver 为该用户 ID,并且 status 为 “accepted” 的记录。

// 获取好友列表的路由router.get("/users/:userId/friends", async (req, res) => {  try {    const userId = req.params.userId;    const friendRequests = await FriendRequest.find({      $or: [{ sender: userId }, { receiver: userId }],      status: "accepted",    }).populate('sender receiver'); // populate 用于获取用户详细信息    const friends = friendRequests.map(request => {      if (request.sender._id.toString() === userId) {        return request.receiver;      } else {        return request.sender;      }    });    res.json(friends);  } catch (error) {    console.error(error);    res.status(500).send("Error fetching friends");  }});

优点:

避免了数据冗余和潜在的数据一致性问题。简化了数据更新的逻辑。

缺点:

每次获取好友列表都需要执行数据库查询,可能会影响性能。

总结

选择哪种方法取决于具体的应用场景。 如果性能是关键因素,并且好友列表不经常更新,那么在 User 模型中维护好友列表可能更合适。 但是,如果数据一致性是首要考虑因素,或者好友列表经常更新,那么通过查询 FriendRequest 模型获取好友列表可能更合适。

无论选择哪种方法,都应该注意潜在的数据一致性问题,并使用事务来确保数据完整性。 此外,应该对代码进行充分的错误处理,以防止意外情况发生。

以上就是在 Mongoose 中实现用户添加好友功能:最佳实践指南的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1524218.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 16:36:44
下一篇 2025年12月20日 16:37:00

相关推荐

  • 如何在用户模式中向好友数组添加用户

    本文旨在指导开发者如何在用户接受好友请求后,将其信息添加到对方用户模式的 friends 数组中。文章将讨论避免前端篡改用户ID的措施,并通过 FriendRequest 模型查询好友列表,同时也会介绍如何在接受好友请求时更新用户模式,并强调使用事务以保证数据一致性的重要性。 确保用户ID的安全性 …

    2025年12月20日
    000
  • 在 Mongoose 中实现用户添加好友功能

    本文旨在指导开发者如何在 Mongoose 中实现用户添加好友的功能,并探讨了维护用户好友关系的不同策略。重点介绍如何安全地处理好友请求,以及在用户接受好友请求后,如何正确更新用户模式中的好友数组。同时,也讨论了使用额外数组存储好友关系的必要性,并提出了更优的查询方案。 安全地处理好友请求 在处理好…

    2025年12月20日
    000
  • JavaScript中防止函数推入数组时立即执行

    本文旨在解决JavaScript中将函数推入数组时函数立即执行的问题。通过将函数包装成匿名函数,可以延迟函数的执行,直到使用Promise.all()等方法需要执行它们时才真正调用。本文将提供详细的示例代码,演示如何避免函数立即执行,并确保它们仅在需要时才被调用。 在JavaScript中,当我们将…

    2025年12月20日
    000
  • JavaScript 中如何避免函数在推入数组时立即执行

    本文旨在解决 JavaScript 中函数被推入数组时立即执行的问题。通过将函数包装在匿名函数中,可以延迟函数的执行,确保函数仅在需要时(例如使用 Promise.all())才被调用。本文将提供详细的示例代码和解释,帮助开发者理解和应用这种技术,从而更有效地管理异步操作。 在 JavaScript…

    2025年12月20日
    000
  • JavaScript 中避免函数推入数组时立即执行

    本文旨在解决 JavaScript 中函数推入数组时立即执行的问题。通过将函数引用而非函数调用推入数组,并结合 Promise.all() 方法,可以实现函数的延迟执行,从而更好地控制异步任务的执行时机。本文将提供详细的示例代码和解释,帮助读者理解和应用这一技巧。 在 JavaScript 中,当我…

    2025年12月20日
    000
  • 使用 Shiny 和 Sortable 创建可滚动 Bucket List

    本文将指导你如何使用 Shiny 和 Sortable.js 库创建一个具有固定高度和滚动条的 bucket list。通过添加 CSS 样式来限制容器高度,并设置 overflow 属性,即可实现当列表项过多时,在容器右侧显示滚动条的效果。 实现可滚动 Bucket List 的步骤 以下步骤将详…

    2025年12月20日
    000
  • 使用 Mongoose 更新用户的好友列表:最佳实践指南

    本文旨在指导开发者如何在使用 Mongoose 构建社交应用时,正确地更新用户的好友列表。文章将探讨如何安全有效地处理好友请求的接受流程,并讨论维护用户好友列表的不同策略,包括直接在 User Schema 中维护以及通过查询 FriendRequest Schema 间接获取。同时,本文将强调数据…

    2025年12月20日
    000
  • 解决 ApexCharts 中日期时间轴梯度填充颜色错位问题

    本文针对 ApexCharts 中在使用 axistype-datetime 的日期时间轴并应用梯度填充时,颜色错位的问题提供了两种解决方案。第一种方案通过计算每个数据点对应的时间戳来精确控制梯度颜色;第二种方案则利用垂直梯度,并根据Y轴的最大值来定义颜色分界点,从而实现颜色的正确对齐。通过本文的学…

    2025年12月20日
    000
  • JavaScript 中防止函数被立即执行并延迟到 Promise.all 执行

    第一段引用上面的摘要: 本文旨在解决 JavaScript 中将函数推入数组时函数被立即执行的问题,并提供解决方案以确保函数仅在 Promise.all() 执行时才被调用。通过将函数引用推入数组,而非直接调用函数,可以实现延迟执行,从而更好地控制异步操作的执行时机。本文将提供详细的代码示例和解释,…

    2025年12月20日
    000
  • 优化OpenAI API:解决GPT应用中意外代码生成问题

    本教程旨在解决使用OpenAI GPT-3.5 API(如text-davinci-003)时,模型意外生成无关代码的问题。文章强调了选择更适合代码生成任务的模型(如gpt-3.5-turbo或gpt-4)的重要性,并深入探讨了通过优化提示词(Prompt Engineering)来提升模型响应质量…

    2025年12月20日
    000
  • Shiny Sortable列表滚动实现教程

    本教程详细介绍了如何在Shiny应用中使用sortable包创建可滚动的列表(rank_list)。核心解决方案是通过CSS样式属性max-height和overflow-y: auto来控制列表容器的高度和溢出行为,从而在内容超出指定高度时自动显示滚动条。文章提供了完整的Shiny应用示例代码,并…

    2025年12月20日
    000
  • 如何利用 CSS-in-JS 技术动态管理组件的样式与主题?

    使用 CSS-in-JS 可实现组件级动态样式与主题切换,以 styled-components 为例,通过模板字符串和 props 动态设置样式,结合 ThemeProvider 统一管理主题,支持状态感知与运行时主题切换,提升封装性与可维护性。 使用 CSS-in-JS 可以将样式逻辑直接写在 …

    2025年12月20日
    000
  • JavaScript中的可选链(Optional Chaining)和空值合并(Nullish Coalescing)如何提升代码健壮性?

    可选链(?.)能安全访问嵌套属性,避免“Cannot read property of undefined”错误;2. 空值合并(??)仅在值为null或undefined时使用默认值,不干扰0、”、false等合法假值;3. 两者结合如userData?.settings?.level…

    2025年12月20日
    000
  • 如何实现一个符合Promise A+规范的Promise类?

    实现一个符合 Promise A+ 规范的 Promise 类,需定义 pending、fulfilled、rejected 三种不可逆状态,通过 resolve 和 reject 函数改变状态并执行对应回调;then 方法返回新 Promise 实现链式调用,根据当前状态异步执行 onFulfil…

    2025年12月20日
    000
  • 如何理解JavaScript中的深拷贝与浅拷贝?

    浅拷贝复制对象第一层,引用类型共享内存地址,修改嵌套对象会影响原对象;深拷贝递归复制所有层级,完全独立,互不影响。常见浅拷贝方法有Object.assign、扩展运算符,深拷贝可用JSON.parse(JSON.stringify())或_.cloneDeep(),但前者不支持函数、undefine…

    2025年12月20日
    000
  • Shiny 应用中实现可滚动 Sortable 列表的实践指南

    本文详细介绍了如何在 Shiny 应用中创建具有滚动功能的 sortable 列表。通过应用 CSS 样式 max-height 和 overflow-y: auto 到 rank_list 容器,用户可以有效管理内容过多的列表,确保界面整洁且用户体验良好。教程将提供完整的代码示例和详细解释,帮助开…

    2025年12月20日
    000
  • JavaScript 的异步函数 async/await 在底层是如何被转换为生成器执行的?

    async/await并非转换为生成器,而是引擎用类似状态机机制实现异步控制。1. async函数返回Promise,自动管理执行流程;2. await暂停执行并等待Promise解决,底层通过状态机保存上下文和恢复;3. 与生成器不同,async/await由引擎原生支持,无需手动调用next()…

    2025年12月20日
    000
  • 在 Node.js 中,如何利用子进程执行系统命令并处理输出?

    Node.js中通过child_process模块执行系统命令,常用方法有exec、spawn及其同步版本。exec适合获取简单命令的完整输出,但会缓存全部结果,不适用于大量数据;spawn提供流式输出,可实时处理大数据或长时间任务;同步方法execSync和spawnSync会阻塞主线程,仅建议在…

    2025年12月20日
    000
  • Bootstrap DataTables 高效集成与 Ajax 数据加载指南

    本文针对Bootstrap 5环境下DataTables功能异常的问题,深入解析了手动填充HTML表格与DataTables工作机制不兼容的根源。教程核心在于指导读者利用DataTables内置的Ajax数据源功能,通过精简的配置,实现从API动态获取数据、自动渲染表格,并确保排序、搜索等高级功能正…

    2025年12月20日
    000
  • 如何利用JavaScript的Generator函数实现异步流程控制?

    Generator函数通过yield暂停执行,结合Promise和执行器可实现异步流程的同步化写法,如run函数驱动Generator处理Promise,使异步操作链更清晰;虽async/await已成为主流,但Generator在复杂控制场景仍有应用价值。 JavaScript的Generator…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信