”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Laravel SoftDelete:避免唯一约束问题

Laravel SoftDelete:避免唯一约束问题

发布于2024-11-06
浏览:957

Laravel SoftDelete: Avoiding the Unique Constraint Problem

本文源自https://medium.com/@hafiqiqmal93/laravel-softdelete-avoiding-the-unique-constraint-problem-45381d9745a0

如果您已经使用 Laravel 一段时间,特别是当项目涉及数据完整性时,您很可能已经遇到过 SoftDelete 功能。非常有用,因为您可以“删除”记录,而无需真正将它们从数据库中取出。 Laravel 所做的只是添加一个deleted_at 时间戳,以便将其标记为已删除,但仍保留在系统中。保留历史数据固然很好,但它确实引入了一个潜在的棘手问题——当您恢复软删除记录时,唯一约束会发生什么情况?


当您想要恢复数据库中已有的记录(例如唯一的电子邮件或用户名)时,这就会成为问题。 Laravel 只会抛出一个错误并停止进程。幸运的是,有一种简单的方法可以非常干净地避免这个问题。

让我们使用一个特性来演示一个解决方案,该特性将帮助您绕过在 Laravel 中使用 SoftDelete 时的独特约束。

了解问题

让我们举一个基本的例子。假设您有一个用户表,其中的电子邮件字段必须是唯一的:


Schema::create('users', function (Blueprint $table) {
    $table->string('email')->unique();
    $table->softDeletes();
});


如果您软删除电子邮件地址为 [email protected] 的用户,然后使用同一电子邮件地址创建一个新用户,Laravel 会抱怨电子邮件字段的唯一约束,并引发错误。在同样的情况下,当你尝试恢复已删除的用户时,Laravel 也会抱怨电子邮件字段的唯一约束并抛出相同的错误。

这变得令人头痛,尤其是在处理记录恢复是一项常见任务的大型系统时。

为了防止这种情况,我们可以在软删除记录时临时更改唯一字段的值,并在恢复记录时恢复原始值。这样,数据库在软删除或恢复期间就不会超出唯一约束。

解决方案:使用特征来避免重复约束

Laravel 特征是封装此功能的好方法。这是我们可以用来处理问题的一个特征:


trashed()) {
                foreach ($model->getDuplicateAvoidColumns() as $column) {
                    // handle for Spatie Translatable library
                    if (method_exists($model, 'getTranslatableAttributes')) {
                        $translates = $model->getTranslatableAttributes();
                        if (in_array($column, $translates)) {
                            foreach ($translates as $translate) {
                                if ($translate === $column) {
                                    $values = $model->getTranslations($column);
                                    foreach ($values as $translation => $value) {
                                        $values[$translation] = (explode('--', $value)[1] ?? $value);
                                    }
                                    $model->setTranslations($column, $values);
                                    break;
                                }
                            }
                            continue;
                        }
                    }
                    if ($value = (explode('--', $model->{$column})[1] ?? null)) {
                        $model->{$column} = $value;
                    }
                }
            }
        });
        static::deleted(function ($model): void {
            foreach ($model->getDuplicateAvoidColumns() as $column) {
                // handle for Spatie Translatable library
                if (method_exists($model, 'getTranslatableAttributes')) {
                    $translates = $model->getTranslatableAttributes();
                    if (in_array($column, $translates)) {
                        foreach ($translates as $translate) {
                            if ($translate === $column) {
                                $values = $model->getTranslations($column);
                                foreach ($values as $translation => $value) {
                                    $values[$translation] = time() . '--' . $value;
                                }
                                $model->setTranslations($column, $values);
                                break;
                            }
                        }
                        continue;
                    }
                }
                $model->{$column} = time() . '--' . $model->{$column};
            }
            $model->save();
        });
    }
}


这个特质有几件事:

  • 删除时,它将时间戳附加到唯一字段,本质上使该字段再次唯一,而不影响原始值。此技巧对于在软删除记录时保持满足唯一约束很有用。
  • 恢复时,它会去除时间戳,恢复唯一字段的原始值。

运作原理:

  • 唯一字段处理:每当删除具有此特征的模型时,该特征都会将时间戳附加到您想要保持唯一的字段(例如:电子邮件、用户名)。如果您尝试添加具有相同唯一值的新记录,这可以防止冲突。
  • 处理可翻译字段: 如果您的模型使用 Spatie Translatable 库,则此特征也足够智能,可以处理多语言字段。它查找可翻译的属性,调整它们的值,并使用时间戳技巧保存它们。
  • 恢复:当您恢复软删除的记录时,该特征会从唯一字段中去除时间戳,将字段返回到其原始值。

将特征应用到您的模型中

以下是如何在 Laravel 模型中应用此特征的方法:


class User extends Model
{
    use SoftDeletes, AvoidDuplicateConstraintSoftDelete;
    // Specify which columns should avoid the unique constraint issue
    public function getDuplicateAvoidColumns(): array
    {
        return ['email', 'username'];
    }
}


通过将 **AvoidDuplicateConstraintSoftDelete** 特征添加到模型中并指定哪些列需要避免唯一约束冲突(例如 emailusername),您可以轻松防止这些问题.

为什么这种方法有效?

这意味着,在软删除记录的情况下,由于某些独特的约束,不会导致与进一步操作的冲突。或者,换句话说,通过这种方式,您将能够通过将时间戳附加到唯一字段,使数据库的记录在唯一性方面“隐藏”,但在需要时仍然可以恢复。

当您处理大型数据库并且记录恢复很常见时,这非常有用。每当您引入软删除用户或任何其他模型时,您不必每次都处理“重复条目”错误。


最后的想法

Laravel 中最有用的东西是 SoftDelete,但有时它在处理独特约束时会让人头疼。这里有一个简单的、基于特征的解决方案,它将提供一种避免问题的优雅方法,只需在删除和恢复时临时更改唯一字段即可。这样您将避免令人沮丧的错误,并让您的应用程序顺利工作,而不会破坏数据库中的唯一约束。

如果您的任何领域已实现多语言化或使用 Spatie 的 Translatable 等库,则上述解决方案在每种情况下都可以正常工作。 SoftDeletes 旨在为您提供灵活性,而不是妨碍您。通过上述小修复,您将避免大多数陷阱并保持数据整洁和用户满意。


通过将此特征添加到您的模型中,您将节省时间并减少麻烦,特别是当您处理软删除和恢复是频繁操作的大型数据集时。在你的 Laravel 项目中尝试一下,你会发现它如何顺利地处理那些棘手的独特约束问题!


感谢您的阅读!不要忘记订阅以随时了解系统设计和电子商务创新的最新动态。设计愉快!

如果您发现本文富有洞察力,并希望了解有关系统设计和技术趋势的更多内容,请务必关注我:-

推特:https://twitter.com/hafiqdotcom
领英:https://www.linkedin.com/in/hafiq93
给我买咖啡:https://paypal.me/mhi9388 /
https://buymeacoffee.com/mhitech
媒体:https://medium.com/@hafiqiqmal93

版本声明 本文转载于:https://dev.to/afiqiqmal/laravel-softdelete-avoiding-the-unique-constraint-problem-8k2?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何使用预先导入的包启动 python 或 ipython。
    如何使用预先导入的包启动 python 或 ipython。
    不必每次执行 python 或 ipython 时都必须编写“import os”或其他常见包,此脚本将启动 python 并导入这些包 #!/usr/bin/env bash # -----------------------------------------------------------...
    编程 发布于2024-11-06
  • python中的数据类型
    python中的数据类型
    内置数据类型 文本类型:字符串 数值类型:int、float、complex 序列类型:列表、元组、范围 映射类型:字典 集合类型:集合、冻结集合 布尔类型:bool 二进制类型:字节、字节数组、内存视图 无类型:无类型 程序 a=“阿维纳什” 打印(类型(a)) b=45 打印(类型(b)) c=...
    编程 发布于2024-11-06
  • JavaScript 中的提升
    JavaScript 中的提升
    什么是 JavaScript 提升? 提升是指 JavaScript 中在执行之前进行声明的过程。首先处理变量和函数声明。因此,即使变量在声明之前被引用,也不会导致错误,而是返回 undefined。对于函数声明,整个函数被提升,这意味着它可以在代码中定义之前使用。此过程在执行开始...
    编程 发布于2024-11-06
  • 如何在 Go 中同时选择缓冲发送和无缓冲接收通道?
    如何在 Go 中同时选择缓冲发送和无缓冲接收通道?
    同时选择 Go 发送和接收通道在 Go 中,可以使用 select 语句在通道上执行非阻塞 I/O 操作。然而,在处理缓冲发送通道 (chan
    编程 发布于2024-11-06
  • 如何将列表列表转换为统一的 NumPy 数组?
    如何将列表列表转换为统一的 NumPy 数组?
    将列表列表转换为 NumPy 数组数据分析中的一个常见任务是将列表列表转换为 NumPy 数组高效的数值运算。该数组可以通过将每个列表分配给一行来形成,列表中的每个元素占据一列。选项 1:数组数组如果子列表具有不同的长度,合适的方法是创建数组的数组。这保留了列表列表的原始结构,从而可以轻松检索特定元...
    编程 发布于2024-11-06
  • 前端的顶级设计模式
    前端的顶级设计模式
    在过去的几个月里,我为前端开发人员分享了一些流行的设计模式。其中包括 Singleton、Facade、Observer、Publisher/Subscriber 等模式。今天,我想总结一下这些模式的一些要点和好处,以及如何使用它们来改进您的前端开发流程。 什么是设计模式 设计模式是...
    编程 发布于2024-11-06
  • ServBay版本.pdate公告
    ServBay版本.pdate公告
    我们很高兴地宣布新版本 1.4.4 已经到来!让我们来看看新增的备受期待的新功能。 新功能 CA和证书管理: 统一SSL证书管理平台:全新的证书管理平台,旨在简化证书申请和管理流程。 ServBay User CA 和 ServBay Public CA: 引入 Se...
    编程 发布于2024-11-06
  • Spring框架中的控制反转
    Spring框架中的控制反转
    控制反转(IoC)和依赖注入(DI)是Spring框架中的两个基本概念。传统上,对象负责创建和管理它们自己的依赖关系。然而,IoC 通过将对象创建和依赖管理的控制权移交给像 Spring 这样的框架来翻转这一责任。 这种转变有几个优点: 更简单的实现交换:只需对代码库进行最小的更改即可交换不同的实现...
    编程 发布于2024-11-06
  • 使用 React 构建递归文件系统:深入探讨
    使用 React 构建递归文件系统:深入探讨
    简介:在 React 中构建递归文件系统 在现代 Web 开发中,创建交互式动态文件系统是一个常见的要求。无论是管理文档、组织项目还是构建复杂的数据结构,拥有强大的文件系统都至关重要。在这篇博文中,我们将探讨如何在 React 中构建递归文件系统,重点关注可以添加、重命名或删除的嵌...
    编程 发布于2024-11-06
  • SQL 查询速度慢?使用此技术提高应用程序的性能
    SQL 查询速度慢?使用此技术提高应用程序的性能
    挑战 在我的应用程序(React Spring Boot Oracle)中,处理大型数据集导致处理时间极其缓慢。我需要一种解决方案来提高性能而不影响准确性或完整性。 解决方案:NTILE 并行处理 NTILE 是一个功能强大的 SQL 窗口函数,旨在将结果集划分为指...
    编程 发布于2024-11-06
  • 关于测试覆盖率的真相
    关于测试覆盖率的真相
    一个强有力的真理。 看下面这段简单明了的代码: function sum(a, b) { return a b; } 现在,让我们为它编写一些测试: test('sum', () => { expect(sum(1, 2)).toBe(3); expect(s...
    编程 发布于2024-11-06
  • 为什么我的 OpenGL 三角形无法在 Go 中渲染?调查顶点缓冲区问题。
    为什么我的 OpenGL 三角形无法在 Go 中渲染?调查顶点缓冲区问题。
    Go 中的 OpenGL 顶点缓冲区问题在尝试在 Go 中使用 OpenGL 显示三角形时,用户遇到了顶点缓冲区问题缓冲区无法渲染形状。 Go 代码源自教程,但与 C 代码不同的是,它没有产生任何输出。问题原因问题的根本原因位于传递给 vertexAttrib.AttribPointer() 的参数...
    编程 发布于2024-11-06
  • 为什么在 Linux 32 位发行版上的 Go 程序中设置 `ulimit -n` 会导致“参数无效”错误?
    为什么在 Linux 32 位发行版上的 Go 程序中设置 `ulimit -n` 会导致“参数无效”错误?
    如何在 Go 程序中设置 ulimit -n?问题用户尝试在 Go 程序中设置 ulimit -n使用 setrlimit 和 getrlimit 系统调用将其限制在程序内而不是全局。然而,在尝试设置该值时出现错误,提示“参数无效”。解决方案发现该问题是由于 Linux 32 的 Getrlimit...
    编程 发布于2024-11-06
  • 如何在Python中创建无限深度的动态嵌套字典?
    如何在Python中创建无限深度的动态嵌套字典?
    未定义深度的动态嵌套字典在涉及复杂多级数据结构的场景中,经常会遇到变量嵌套字典的需求水平。虽然硬编码插入语句是一种潜在的解决方案,但当事先未知嵌套深度时,这种方法是不切实际的。要克服此限制,请考虑利用 Python 的 collections.defaultdict,它允许动态创建字典。可以使用以下...
    编程 发布于2024-11-06
  • Python 变得强大:轻松编程的初学者指南
    Python 变得强大:轻松编程的初学者指南
    Python 是一门强大的编程语言,语法简单,应用广泛。安装 Python 后,可以学习其基本语法,包括变量赋值、数据类型和流程控制。实战案例中,我们通过蒙特卡罗模拟计算圆周率,展示了 Python 在数值计算中的能力。此外,Python 拥有丰富的库,涵盖机器学习、数据分析和网络开发等领域,体现了...
    编程 发布于2024-11-06

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

Copyright© 2022 湘ICP备2022001581号-3