使用 Django 和 JavaScript 实现流畅的点赞/取消点赞功能

使用 django 和 javascript 实现流畅的点赞/取消点赞功能

本文旨在帮助开发者使用 Django 框架和 JavaScript 实现一个流畅、无需页面刷新的点赞/取消点赞功能。我们将探讨如何正确处理图标切换、避免点赞计数在所有帖子中同步更新的问题,并提供一个更简洁、高效的代码实现方案,包括前后端代码示例和注意事项。

前端实现:JavaScript 和 HTML

首先,我们需要在 HTML 模板中渲染帖子列表,并为每个帖子添加一个点赞/取消点赞按钮。关键在于,在页面初次加载时,根据用户是否已点赞该帖子,正确显示相应的图标。

{% for post in posts %}

{{post.title}}

{{post.body}}

立即学习Java免费学习笔记(深入)”;

{{ post.postLiked.count }} Likes{% endfor %} function getCookie(name) { let cookieValue = null; if (document.cookie && document.cookie !== '') { const cookies = document.cookie.split(';'); for (let i = 0; i { const thumbs_up = ''; const thumbs_down = ''; const csrftoken = getCookie("csrftoken"); const user = "{{request.user.id}}"; const response = await fetch("/post/like/", { method: "POST", headers: { "X-CSRFToken": csrftoken, 'Content-Type': 'application/json' }, body: JSON.stringify({ user_id: user, post_id: id }), }); const data = await response.json(); const like_button_element = document.getElementById( `like_button_${id}` ); const likes_count_element = document.getElementById( `likes_count_${id}` ); let num_likes = parseInt(likes_count_element.innerHTML); if (data.liked) { like_button_element.innerHTML = thumbs_down; likes_count_element.innerHTML = num_likes + 1; } else { like_button_element.innerHTML = thumbs_up; likes_count_element.innerHTML = num_likes - 1; } };

关键点:

唯一ID: 使用 like_button_{{post.id}} 作为按钮的 ID,确保每个按钮都有唯一的标识符。这解决了点赞计数同步更新的问题。初始状态: {% if post.id in user_liked_posts %} 语句根据用户是否已点赞来决定初始显示的图标。getCookie 函数: 用于获取 CSRF token,这是 Django 安全机制的要求,用于防止跨站请求伪造攻击。异步请求: 使用 fetch API 发送异步请求,避免页面刷新。错误处理: 建议在 fetch 中添加错误处理机制,例如使用 try…catch 块捕获网络错误或服务器错误。Content-Type: 在 fetch 的 headers 中设置 Content-Type’: ‘application/json’,确保后端能正确解析 JSON 数据。点赞计数显示: 添加了 span 元素,用于显示每个帖子的点赞数,并使用唯一的 ID likes_count_{{post.id}}。

后端实现:Django Views 和 Models

在 Django 的 views.py 中,我们需要编写两个函数:一个用于渲染帖子列表,另一个用于处理点赞/取消点赞的请求。

from django.shortcuts import renderfrom django.contrib.auth import get_user_modelfrom django.http import JsonResponsefrom .models import Post, LikeOrUnlikeimport jsondef list_posts(request):    posts = Post.objects.all()    user_liked_posts = request.user.likeByUser.all().values_list("post", flat=True)    context = {"posts": posts, "user_liked_posts": user_liked_posts}    return render(request, "posts/list.html", context)def like_or_unlike_fetch(request):    data = json.loads(request.body)    user_id = data.pop("user_id")    post_id = data.pop("post_id")    user = get_user_model().objects.get(pk=user_id)    post = Post.objects.get(pk=post_id)    data.update({"user": user, "post": post})    like_object = LikeOrUnlike.objects.filter(**data)    if not like_object.exists():        LikeOrUnlike.objects.create(**data)        liked = True    else:        like_object.delete()        liked = False    return JsonResponse({"liked": liked})

models.py:

from django.db import modelsfrom django.contrib.auth.models import Userclass Post(models.Model):    title = models.CharField(max_length=200)    body = models.TextField()    def __str__(self):        return self.titleclass LikeOrUnlike(models.Model):    user = models.ForeignKey(User, on_delete=models.CASCADE, related_name='likeByUser')    post = models.ForeignKey(Post, on_delete=models.CASCADE, related_name='postLiked')    def __str__(self):        return f'{self.user} liked this {self.post}'

urls.py:

from django.urls import pathfrom . import viewsurlpatterns = [    path('posts/', views.list_posts, name='list_posts'),    path('post/like/', views.like_or_unlike_fetch, name='like_or_unlike_fetch'),]

关键点:

list_posts 函数: 获取所有帖子和当前用户已点赞的帖子,并将它们传递给模板。user.likeByUser.all().values_list(“post”, flat=True) 使用反向关联查询用户点赞的所有帖子ID。like_or_unlike_fetch 函数: 接收来自前端的异步请求,根据用户是否已点赞该帖子来创建或删除 LikeOrUnlike 对象。JSON 处理: 使用 json.loads(request.body) 解析前端发送的 JSON 数据。get_user_model(): 使用 get_user_model() 获取用户模型,避免硬编码 User 模型。exists(): 使用 .exists() 检查是否存在点赞记录,提高效率。原子性操作: 点赞和取消点赞操作应该具有原子性。如果可能,考虑使用事务来确保数据一致性。异常处理: 在 like_or_unlike_fetch 函数中添加异常处理,例如处理用户或帖子不存在的情况。

总结

通过上述步骤,我们可以使用 Django 和 JavaScript 实现一个流畅、无需页面刷新的点赞/取消点赞功能。关键在于正确处理前端的图标切换、避免点赞计数同步更新的问题,并使用简洁、高效的代码实现方案。 记住,安全性是至关重要的,始终要验证用户输入并防止 CSRF 攻击。 此外,可以考虑使用缓存来提高性能,特别是在点赞数量非常大的情况下。

以上就是使用 Django 和 JavaScript 实现流畅的点赞/取消点赞功能的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 15:56:18
下一篇 2025年12月18日 15:51:00

相关推荐

  • 使用 Django 和 JavaScript 实现平滑的点赞/取消点赞功能

    本文旨在提供一个清晰、高效的教程,指导开发者如何使用 Django 框架和 JavaScript 实现点赞和取消点赞功能,并解决常见问题,例如图标切换不正确和点赞计数错误。我们将优化数据处理方式,采用更简洁的 JavaScript 代码,并提供完整的示例,确保功能的流畅性和用户体验。 优化模型关系 …

    2025年12月20日 好文分享
    000
  • JavaScript:访问具有相同类名的表单内的元素

    本文旨在解决在包含多个表单的网页中,如何使用 JavaScript 准确访问特定表单内的元素,尤其是当不同表单中的元素具有相同的类名时。文章将通过示例代码详细讲解如何通过表单的 ID 来定位表单,并获取表单内特定类名的元素的值。同时,也会介绍一些调试技巧,帮助开发者更好地理解和排查问题。 在构建复杂…

    2025年12月20日 好文分享
    000
  • 优化 React 组件中的条件判断:减少 if-else 语句的使用

    本文旨在帮助开发者优化 React 组件中大量 if-else 语句的使用,通过采用更简洁的数据结构和 JavaScript 特性,如对象字面量和 ??= 空值合并赋值运算符,来提高代码的可读性和可维护性。我们将通过一个实际的 React 组件案例,展示如何将复杂的条件判断逻辑简化为更优雅的实现方式…

    2025年12月20日
    000
  • 优化 React 代码中的 If-Else 结构:更简洁的条件处理方案

    本文旨在帮助开发者优化React代码中大量的if-else语句,提供使用对象字面量和三元运算符等方法,以实现更简洁、更易于维护的代码。通过具体的代码示例,展示如何将复杂的条件判断逻辑转换为更优雅的实现方式,提高代码的可读性和效率。 在编写 React 应用时,经常会遇到需要根据不同的条件执行不同逻辑…

    2025年12月20日
    000
  • 网页表单验证:确保必填字段完整性的 JavaScript 教程

    本文旨在解决 JavaScript 表单验证中常见的逻辑错误,并提供一种简单有效的验证方法。通过修改原有的验证逻辑,避免了不必要的复杂判断,简化代码,并确保在必填字段为空时阻止表单提交,从而提高用户体验和数据质量。本文将提供修正后的代码示例,并解释其工作原理,帮助开发者更好地理解和应用表单验证技术。…

    好文分享 2025年12月20日
    000
  • Prisma Client Extensions 中异步计算字段的解决方案

    Prisma Client Extensions 的 result 扩展旨在提供同步计算字段,直接在 compute 函数中使用 await 会导致 Promise 对象被返回或挂起。本文将深入探讨这一限制,并提供两种有效的解决方案:一是让 compute 函数返回一个异步函数,将异步操作延迟到实际…

    2025年12月20日
    000
  • 在富文本编辑器中,如何利用 JavaScript 处理选区范围和内容格式?

    答案:JavaScript通过Selection和Range API实现富文本编辑功能。使用window.getSelection()获取选区,range.surroundContents()包裹内容实现加粗等格式,注意跨节点选区需先extractContents();插入内容时用insertNod…

    2025年12月20日
    000
  • 如何利用 JavaScript 实现一个支持快捷键操作的富文本编辑器?

    答案:通过contenteditable结合keydown事件监听实现快捷键操作,使用document.execCommand执行加粗、斜体、下划线、插入链接等命令,支持Ctrl/Cmd组合键触发,可扩展对齐、列表、标题等功能,并建议后续迁移至现代API或使用专业库优化兼容性与功能。 要实现一个支持…

    2025年12月20日
    000
  • Redux Dispatch 无效:状态未更新的调试与解决方案

    本文旨在帮助开发者诊断和解决 Redux 应用中 dispatch action 后状态未更新的问题。通过分析常见的错误配置和代码陷阱,提供逐步排查和修复的指导,确保 Redux 状态管理的正确性和可靠性。本文将通过一个实际案例,深入剖析问题原因,并提供详细的解决方案,助力开发者构建稳定高效的 Re…

    2025年12月20日
    000
  • Redux Dispatch 未更新 State 的问题排查与解决

    本文旨在帮助开发者排查和解决 Redux dispatch 未能正确更新 state 的问题。通过分析常见原因,例如 reducer 中的 state 访问错误、dispatch 参数错误等,并提供相应的代码示例和调试技巧,确保 Redux 状态管理的正确性。本文将通过一个实际案例,深入探讨问题根源…

    2025年12月20日
    000
  • Vue.js中JSON数据邮件地址搜索与表格展示教程

    本教程详细介绍了如何在Vue.js应用中,对JSON数据源进行邮件地址搜索。我们将利用JavaScript的Array.prototype.find()方法高效定位匹配的邮件地址对象,并将其结果动态绑定到Vue.js的网格表格中,实现数据的筛选与展示。 理解JSON数据中的邮件地址搜索需求 在现代w…

    2025年12月20日
    000
  • 如何在 ReactJS 18 的类组件中访问 props.children

    本文针对 ReactJS 18 中类组件访问 props.children 时可能遇到的 TypeScript 类型问题,提供了清晰的解决方案。通过正确定义组件的 Props 类型,显式声明 children 属性,可以避免类型检查错误,并确保在类组件中正常使用 props.children。文章还…

    2025年12月20日
    000
  • Redux Dispatch 无效:状态未更新问题排查与解决方案

    本文旨在解决 Redux 应用中 dispatch 函数调用后状态未更新的问题。通过分析常见原因,例如 reducer 中的状态更新方式错误,以及 dispatch 调用时传递的参数不正确等,提供详细的排查步骤和解决方案,帮助开发者快速定位并修复问题,确保 Redux 状态管理的正确性。 在 Red…

    2025年12月20日
    000
  • Vue.js中实现JSON数据电子邮件搜索及表格展示教程

    本教程将指导您如何在Vue.%ignore_a_1%应用中高效地搜索存储在JSON对象数组中的特定电子邮件地址,并将匹配到的数据精准地呈现在网格表格中。我们将利用JavaScript的Array.prototype.find()方法,结合Vue的数据响应式机制,实现一个简洁且功能完善的搜索功能。 1…

    2025年12月20日
    000
  • JavaScript实现表单提交前用户确认与阻止

    本教程详细讲解如何使用JavaScript在用户提交表单前弹出确认对话框。我们将探讨如何正确监听表单的submit事件,并利用confirm()函数获取用户选择。当用户选择取消时,通过event.preventDefault()方法有效阻止表单的默认提交行为,确保数据提交的准确性和用户体验。 1. …

    2025年12月20日
    000
  • Redux Dispatch 不更新状态问题排查与解决

    本文旨在帮助开发者解决 Redux 中 dispatch 无法更新状态的问题。通过分析常见的错误原因,例如 reducer 中的状态更新方式,以及 action payload 的传递,提供详细的排查步骤和解决方案,确保 Redux 状态的正确更新。 当你在 React 应用中使用 Redux 时,…

    2025年12月20日 好文分享
    000
  • Redux Dispatch 不更新状态的排查与解决

    本文旨在帮助开发者诊断和解决 Redux 应用中 dispatch 函数调用后状态未更新的问题。通过分析常见的错误配置和代码实现,提供逐步排查方案和修正建议,确保 Redux 状态管理的正确性和可靠性。 在 Redux 应用开发中,dispatch 函数用于触发状态变更,如果 dispatch 调用…

    2025年12月20日
    000
  • 在Vue.js中实现JSON对象邮箱地址搜索并展示到表格

    本文详细介绍了如何在Vue.js应用中,高效地从JSON数据集合中搜索特定邮箱地址,并将匹配到的单一用户数据展示到前端表格。核心方法是利用JavaScript的Array.prototype.find()函数进行精准查找,并提供了详细的Vue.js代码示例、处理多结果的filter()方法,以及关于…

    2025年12月20日
    000
  • 使用 Sequelize 进行关联查询时,排序混乱问题排查与解决

    本文旨在解决在使用 Sequelize 进行关联查询时,由于关联表数据未排序,导致主表数据排序出现混乱的问题。我们将深入探讨问题产生的原因,并提供一种通过指定更明确的排序规则来解决该问题的方案,确保数据在每次查询时都以一致的顺序返回。 问题分析 在使用 Sequelize 进行关联查询时,特别是涉及…

    2025年12月20日
    000
  • 基于 Sequelize 的多条件排序导致 UI 刷新异常的解决方案

    在使用 Sequelize 进行数据查询时,多条件排序可能会导致 UI 刷新出现异常,例如数据顺序错乱。这是因为在包含关联模型且关联模型也参与排序的情况下,如果关联模型的数据没有明确的排序规则,Sequelize 的查询结果可能是不稳定的。本文将深入探讨这一问题,分析其产生的原因,并提供一种解决方案…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信