”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Python 和 Django 中的 CRUD 操作 - 第 2 部分

Python 和 Django 中的 CRUD 操作 - 第 2 部分

发布于2024-07-29
浏览:937

在上一篇文章中,我们介绍了设置 Django 项目的基础知识并创建了练习模型,我们将其以列表的形式显示在前端。在本文中,我们将深入探讨执行 CRUD 操作。对于那些不熟悉的人来说,CRUD 代表创建、读取、更新和删除 — 本质上是您可以对数据执行的四个基本操作。

现在我们已经在应用文件夹中设置了 API,我们只需扩展索引视图来处理创建、更新和删除请求。

表格

让我们设置一个允许用户创建练习的表单。我们将再次使用 HTML 模板来实现此目的。首先,在 app/templates 文件夹中创建一个名为 add_exercise.html 的新模板。

{% csrf_token %}

接下来,在我们的 index.html 模板中,我们将使用以下方法包含 add_exercise.html 模板:

{% extends "base.html" %} {% block content %}
    

Exercises

{% include 'add_exercise.html' %} ... {% endblock %}

我们在这里利用 include 标签,它可以提高 HTML 模板的可组合性,使我们的代码更易于维护和理解。如果您在浏览器中刷新页面,您应该会看到该表单出现在屏幕上。

Add Exercise

在我们的 HTML 中,我们使用

标记,其方法属性设置为 POST,操作属性指向 /,这与我们用来获取练习列表的端点相同。

在此上下文中,csrf_token 是由随机生成的秘密值表示的安全功能。它有助于保护我们提交的表单免受伪造攻击,这就是 CSRF 的意思——跨站请求伪造。为每个用户会话生成一个唯一的令牌,第三方站点无法访问该令牌,从而防止发生未经授权的更改。

我们的表单包含两个输入字段:一个用于标题,另一个用于日期,遵循我们的练习模型的架构。提交表单后,标题和日期的值将通过 POST 请求发送到 / 端点,然后由 app/views.py 中的索引视图进行处理。

该模型

在 Django 中,我们可以通过添加与 CRUD 操作相对应的特定方法来增强我们的 Exercise 模型(本质上是一个 Python 类)。在 app/models.py 文件中,我们将包含以下内容:

class Exercise(models.Model):
    ...

    def create(request):
        title = request.POST.get('title')
        date = request.POST.get('date')

        exercise = Exercise.objects.create(title=title, date=date)

        return exercise

我们可以从 POST 请求访问标题和日期,如上面的代码所示。然后,我们可以利用Django内置的ORM来创建一个新的练习并返回创建的实例。

我们将利用用于检索练习的相同索引视图,对其进行扩展以检查请求方法是否为 POST。如果是这样,我们将把请求对象传递给我们之前定义的类方法。创建练习后,我们会将用户重定向回主页或执行页面刷新,确保新添加的练习出现在屏幕上。

from django.http import HttpResponseRedirect

from app import models

...

def index(request):
    if request.method == 'POST':
        models.Exercise.create(request)
        return redirect('/')

    exercises = (
        models.Exercise.objects.all().order_by("created_at")
    )
    return render(request, "index.html", context={'exercises': exercises})

现在尝试创建一个新练习,您应该会看到它出现在列表底部。

更新练习

在向练习中添加更新功能之前,让我们先重构一下代码。我们将把练习移至他们自己的模板,名为exercise.html。

Exercises

{% include 'add_exercise.html' %}
    {% for exercise in exercises %}
  • {% include 'exercise.html' %}
  • {% endfor %}

在 app/templates 文件夹中为 exercise.html 创建一个模板,我们将向其中添加以下 HTML:

{% csrf_token %}

我们再次对列表中的每个练习使用

标签,并为 exercise.id 添加隐藏输入,该输入将用于更新练习。返回浏览器并刷新页面;您应该会看到列表中每个练习的表格,每个输入都预先填充了相应的练习数据。

CRUD Operations In Python & Django - Part 2

请注意,我们没有使用 PUT 作为表单方法;相反,我们使用 POST。这是因为视图处理程序只能解析通过 GET 和 POST 请求发送的数据,没有对 PUT 和 DELETE 的内置支持。当我们在 Exercise 类中创建 create 类方法时,您可能已经注意到我们使用了 request.POST.get('title')。虽然这适用于 POST 请求,但请求对象中没有可用的 PUT 或 DELETE 方法。

但是我们如何区分 POST 和 PUT 请求呢?如果您检查我们之前创建的表单,您会注意到我们为提交按钮分配了一个名称属性。我们可以像访问标题和日期一样访问此属性,使用 request.POST.get('update').

让我们更新创建练习表单以包含相同的更改。

...

在我们的练习视图中,我们将进行以下更改以区分请求。

def index(request):
    if request.method == 'POST':
        create = 'create' in request.POST
        update = 'update' in request.POST

        if create == True:
            models.Exercise.create(request)
        elif update == True:
            models.Exercise.update(request)

        return redirect('/')

    exercises = (
        models.Exercise.objects.all().order_by("created_at")
    )
    return render(request, "index.html", context={'exercises': exercises})

我们检查按钮名称并相应地将请求转发到适当的练习方法。

让我们向 app/models.py 中的 Exercise 模型添加更新类方法。

def update(request):
    id = request.POST.get('id')
    title = request.POST.get('title')
    date = request.POST.get('date')

    exercise = Exercise.objects.filter(pk=id).update(title=title, date=date)

    return exercise

要更新数据库中的行,我们可以使用 Exercise 模型上提供的更新方法。然而,在更新之前,我们需要确保我们更新的是正确的练习。为此,我们按主键 id 过滤练习,并仅更新该特定练习。

删除练习

同样,我们将在exercise.html模板中的每个练习旁边添加一个删除按钮。

...

我们将把delete设置为name属性的值,并在views.py中,我们将扩展if...elif语句来处理删除操作。

def index(request):
    if request.method == 'POST':
        create = 'create' in request.POST
        update = 'update' in request.POST
        delete = 'delete' in request.POST

        if create == True:
            models.Exercise.create(request)
        elif update == True:
            models.Exercise.update(request)
        elif delete == True:
            models.Exercise.delete(request)

        return redirect('/')

    exercises = (
        models.Exercise.objects.all().order_by("created_at")
    )
    return render(request, "index.html", context={'exercises': exercises})

在练习模型中,我们将添加类方法delete。

def delete(request):
    id = request.POST.get('id')
    is_deleted = Exercise.objects.filter(pk=id).delete()

    if is_deleted == 1:
        return True

    return False

通过此添加,我们已在 Python 和 Django 应用程序中成功实现了 CRUD 操作。

要点

  1. Django 视图处理程序不支持 PUT 和 DELETE 请求,因为它们不解析这些 HTTP 方法的查询参数或请求正文。因此,我们必须依赖 POST 请求,并通过在请求正文中传递附加字段来区分它们。
  2. 注意到我正在向我从中获取练习的同一路线发出 POST 请求。这很重要,因为如果您要创建像 /api/exercises 这样的端点来处理请求,则需要手动管理重定向。否则,请求后
    标记的行为是将用户重定向到action属性中指定的端点。因此,您需要手动将用户重定向回所需页面,或者在我们的示例中,将用户保持在同一页面上。
from django.http import HttpResponseRedirect

def index(request):
    ...

    return redirect('/')
    # or
    return HttpResponseRedirect(request.META['HTTP_REFERER'])

总之,通过有效管理我们的 POST 请求并确保正确的重定向,我们可以在 Django 应用程序中实现 CRUD 操作的同时创建无缝的用户体验。

版本声明 本文转载于:https://dev.to/prvnbist/crud-operations-in-python-django-part-2-4fad?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-11-23
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-11-23
  • 尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    解决 PHP 中的 POST 请求故障在提供的代码片段中:action=''而不是:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"检查 $_POST数组:表单提交后使用 var_dump 检查 $_POST 数...
    编程 发布于2024-11-23
  • 如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    在 PHP 中组合关联数组在 PHP 中,将两个关联数组组合成一个数组是一项常见任务。考虑以下请求:问题描述:提供的代码定义了两个关联数组,$array1 和 $array2。目标是创建一个新数组 $array3,它合并两个数组中的所有键值对。 此外,提供的数组具有唯一的 ID,而名称可能重合。要求...
    编程 发布于2024-11-23
  • 在 Go 中使用 WebSocket 进行实时通信
    在 Go 中使用 WebSocket 进行实时通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要一种比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSoc...
    编程 发布于2024-11-23
  • 大批
    大批
    方法是可以在对象上调用的 fns 数组是对象,因此它们在 JS 中也有方法。 slice(begin):将数组的一部分提取到新数组中,而不改变原始数组。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index p...
    编程 发布于2024-11-23
  • Python 和 JSON 中单引号和双引号可以互换使用吗?
    Python 和 JSON 中单引号和双引号可以互换使用吗?
    Python 中单引号和双引号的互换性在 Python 编程中,单引号和双引号在定义字符串时通常可以互换。然而,这种互换性并没有扩展到 JSON(JavaScript 对象表示法),它对字符串语法有严格的要求。JSON 要求所有字符串值都用双引号引起来。在 JSON 字符串中使用单引号是不正确的,会...
    编程 发布于2024-11-23
  • 如何在命令行程序的 JUnit 测试中模拟用户输入?
    如何在命令行程序的 JUnit 测试中模拟用户输入?
    使用 JUnit 进行测试模拟系统在软件测试领域,人们在处理以下问题时经常面临模拟用户输入的挑战命令行程序。当程序提示通过 System.in 输入时,如何在 JUnit 测试中自动执行此行为?解决方案绕过 System.in 并注入模拟用户输入,按照以下步骤操作:建立抽象层:避免在内部直接调用 S...
    编程 发布于2024-11-23
  • 如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 识别今天生日的用户使用 MySQL 确定今天是否是用户的生日涉及查找生日匹配的所有行今天的日期。这可以通过一个简单的 MySQL 查询来实现,该查询将存储为 UNIX 时间戳的生日与今天的日期进行比较。以下 SQL 查询将获取今天有生日的所有用户: FROM USERS ...
    编程 发布于2024-11-23
  • 如何使用 Java CSS 解析器提取 HTML 文档中特定元素的 CSS 样式?
    如何使用 Java CSS 解析器提取 HTML 文档中特定元素的 CSS 样式?
    Java CSS 解析器要求目标是使用 Java CSS 解析器获取 HTML 文档中特定元素的 CSS 样式。解决方案CSSParser推荐一个选项是 CSSParser,一个提供错误反馈的强大解析器。下面是它的用法示例:import com.steadystate.css.parser.CSSO...
    编程 发布于2024-11-23
  • 除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-11-23
  • 如何检索 Python 请求中的重定向历史记录?
    如何检索 Python 请求中的重定向历史记录?
    在Python请求中重定向URL在Python的Requests库中,设置allow_redirects=True允许库自动遵循HTTP重定向。但是,它不提供在重定向后检索新 URL 的直接方法。要访问重定向历史记录,您可以利用 response.history 属性。此属性包含一个 Respons...
    编程 发布于2024-11-23
  • 为什么我应该使用“pip install --user”?
    为什么我应该使用“pip install --user”?
    理解“pip install --user ...”的目的:简化包安装Pip,一个强大的 Python 包管理器,允许您安装来自各种来源的软件包。默认情况下,pip 将软件包安装到系统范围的目录中,通常需要提升权限(root 访问权限)。然而,--user 标志为那些喜欢安装软件包而不需要特殊权限的...
    编程 发布于2024-11-23
  • scanf() 与 cin:C++ 输入哪个更快?
    scanf() 与 cin:C++ 输入哪个更快?
    速度比较:scanf() vs. cin问题:是真的吗在 C 程序中使用 scanf() 比使用cin?答案:是的,scanf() 确实比 C 中的 cin 更快,正如基准测试结果所证明的。要进行在测试中,编写了一个简单的程序来读取大量数字并计算它们的 XOR 值。该程序使用stdio的scanf(...
    编程 发布于2024-11-23
  • 如何在不使用 Boost 的情况下在 C/C++ 中生成正态分布的随机数?
    如何在不使用 Boost 的情况下在 C/C++ 中生成正态分布的随机数?
    在C/C中生成正态分布的随机数问题:如何轻松生成随机数在不使用 Boost 的情况下遵循 C 或 C 的正态分布库?答案:Box-Muller 变换Box-Muller 变换是一种广泛使用的生成方法来自均匀随机数生成器的正态分布数字。它产生的值精确地遵循高斯分布。Box-Muller 变换的数学公式...
    编程 发布于2024-11-23

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3