在Mongoose中实现好友关系:更新User Schema中的好友数组

在mongoose中实现好友关系:更新user schema中的好友数组

本文旨在指导开发者如何在Mongoose中实现用户添加好友的功能,重点讲解在接受好友请求后如何正确更新User Schema中的好友数组。同时,文章也会探讨更高效的数据结构设计方案,以及在使用事务时需要注意的关键点,确保数据一致性。

安全地处理好友请求

首先,确保在处理好友请求时,发送者ID来自已验证的用户会话,而不是直接从前端表单获取。从前端获取用户ID存在安全风险,用户可以通过开发者工具篡改表单数据,冒充他人发送请求。

推荐的做法是从服务器端的会话或身份验证中间件中获取当前用户的ID。以下是修改后的前端表单和后端路由示例:

前端表单 (修改后):

  <input type="hidden" name="receiver" value="" />  

后端路由 (修改后):

router.post("/requests", (req, res) => {  const { receiver } = req.body;  FriendRequest.create({    sender: req.user._id, // 从已验证的用户会话中获取 sender ID    receiver,  })    .then(() => {      res.status(204).send();    })    .catch((error) => {      res.status(500).send("Error sending friend request");    });});

在这个修改后的代码中,req.user._id 代表当前已验证用户的ID,你需要根据你使用的身份验证中间件来调整获取用户ID的方式。

更新User Schema中的好友数组

当用户接受好友请求时,需要更新双方的 User Schema,将对方的 _id 添加到各自的 friends 数组中。以下是更新 User Schema 的代码示例:

router.post("/requests/:id/accept", async (req, res) => {  try {    const request = await FriendRequest.findById(req.params.id);    request.status = "accepted";    await request.save();    const currentUserId = req.user._id; // 获取当前用户ID    const currentUser = await User.findById(currentUserId);    const otherUser = await User.findById(request.sender);    currentUser.friends.push(request.receiver);    otherUser.friends.push(currentUserId);    await currentUser.save();    await otherUser.save();    res.redirect("/requests");  } catch (error) {    console.error("Error accepting friend request:", error);    res.status(500).send("Error accepting friend request");  }});

注意事项:

在更新 friends 数组之前,请确保 request.receiver 和 currentUserId 都是有效的 ObjectId 类型。在生产环境中,需要添加错误处理机制,以便在发生错误时能够及时发现并处理。

使用事务保证数据一致性

直接修改两个不同的集合(FriendRequest 和 User)可能导致数据不一致。例如,如果服务器在更新 FriendRequest 集合后崩溃,但尚未更新 User 集合,则会导致数据不一致。为了解决这个问题,可以使用 Mongoose 的事务功能。

以下是使用事务的代码示例:

const mongoose = require('mongoose');router.post("/requests/:id/accept", async (req, res) => {  const session = await mongoose.startSession();  session.startTransaction();  try {    const request = await FriendRequest.findById(req.params.id).session(session);    request.status = "accepted";    await request.save({ session });    const currentUserId = req.user._id; // 获取当前用户ID    const currentUser = await User.findById(currentUserId).session(session);    const otherUser = await User.findById(request.sender).session(session);    currentUser.friends.push(request.receiver);    otherUser.friends.push(currentUserId);    await currentUser.save({ session });    await otherUser.save({ session });    await session.commitTransaction();    session.endSession();    res.redirect("/requests");  } catch (error) {    await session.abortTransaction();    session.endSession();    console.error("Error accepting friend request:", error);    res.status(500).send("Error accepting friend request");  }});

关键点:

mongoose.startSession() 创建一个新的会话。session.startTransaction() 启动一个事务。在查询和保存文档时,使用 .session(session) 将操作与事务关联。session.commitTransaction() 提交事务,所有操作都将永久保存到数据库。session.abortTransaction() 回滚事务,所有操作都将撤销。session.endSession() 结束会话。

更高效的数据结构设计

虽然在 User Schema 中维护一个 friends 数组是常见的做法,但它可能不是最有效的方法,尤其是在用户拥有大量好友的情况下。

另一种更高效的方法是只维护 FriendRequest 集合,并通过查询来获取好友列表。例如:

FriendRequest.find({  $or: [    { sender: req.user._id, status: "accepted" },    { receiver: req.user._id, status: "accepted" },  ],})  .populate("sender receiver") // 可选:填充用户数据  .then((friendRequests) => {    const friends = friendRequests.map((request) => {      return request.sender._id.equals(req.user._id)        ? request.receiver        : request.sender;    });    console.log(friends);  })  .catch((e) => {    // handle error  });

这种方法的优点是不需要在多个集合中维护数据,从而避免了数据不一致的问题。缺点是每次获取好友列表都需要执行查询。

总结

在 Mongoose 中实现好友关系需要仔细考虑数据一致性和性能。使用事务可以确保数据一致性,但会增加代码的复杂性。选择合适的数据结构取决于具体的应用场景和性能要求。在开发过程中,请始终关注安全性,并避免从前端获取敏感数据

以上就是在Mongoose中实现好友关系:更新User Schema中的好友数组的详细内容,更多请关注创想鸟其它相关文章!

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

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

相关推荐

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

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

    2025年12月20日
    000
  • 如何在用户模式中向好友数组添加用户

    本文旨在指导开发者如何在用户接受好友请求后,将其信息添加到对方用户模式的 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 中的 WeakMap 和 WeakSet 在管理 DOM 节点内存泄漏时有何妙用?

    WeakMap和WeakSet通过弱引用避免内存泄漏,适合存储DOM节点的私有数据或标记已处理节点。使用WeakMap可将元信息与节点关联而不阻止回收,如setNodeData存储状态;WeakSet可用于observeOnce确保事件只绑定一次,节点移除后数据自动释放;此外,WeakMap能解耦闭…

    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

发表回复

登录后才能评论
关注微信