”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 在数据库中更新插入不使用主键或唯一约束的行

在数据库中更新插入不使用主键或唯一约束的行

发布于2024-11-08
浏览:141

Upsert a row in a DB that doesn

在我 7 年的程序员职业生涯中,我大部分时间都是通过 ORM 与 SQL 交互。我发现 Laravel 的 Eloquent ORM 的一项特别有用的功能是它的 updateOrInsert() 方法:

DB::table('posts')
    ->updateOrInsert(
        ['slug' => 'about'], // matching condition
        ['content' => 'Like and subscribe'] // created or updated values
    );

在上面的示例中,Eloquent 将在 posts 表中查找 slug 等于“about”的行。如果存在带有该 slug 的行,Eloquent 会将该行的内容更新为“点赞并订阅”。如果行与该slug存在,Eloquent将创建一个新行,其中slug为“about”,内容为“Like and subscribe”。

问题:没有主键或唯一约束

我目前正在编写一个迁移脚本,将页面数据从旧的 WordPress 站点移动到新的 WordPress 站点。该脚本连接到旧站点的数据库,然后创建一个可以导入到新站点的数据库中的 SQL 文件。新网站已有一些已手动移动的页面,但其内容可能已过时。当新站点上已经存在页面时,我们不想再次创建它:我们想要更新已经存在的页面。我们可以通过使用 WordPress 数据库中的 post_name 列来确定该页面是否已经存在,该列对应于页面的 URL slug。

如果 post_name 是主键,我们可以使用 REPLACE 语句完成类似于 Eloquent 的 createOrUpdate() 方法的操作,但 post_name 不是主键。

如果 post_name 有唯一约束,我们可以使用 INSERT ... ON DUPLICATE KEY UPDATE 语句完成类似于 Eloquent 的 createOrUpdate() 方法的操作,但 post_name 没有唯一约束。

啊,WordPress 的乐趣。

我研究了MySQL的IF语句,但IF语句只适用于存储过程、触发器和函数。我不想在要导入到新站点数据库的 SQL 文件中创建存储过程,因此我必须搜索其他选项。

解决方案:2条语句

我能找到的在纯 SQL 中模拟 Eloquent 的 updateOrInsert() 功能的最简单的解决方案是将问题分成两部分:

  1. 使用匹配的 slug 更新任何现有行。
  2. 如果没有匹配 slugs 的行,则创建一个新行。

实际情况如下:

-- Update the post if it already exists.

UPDATE wp_posts
SET post_type = 'page',
    post_title = 'About',
    post_content = 'Like and subscribe'
WHERE post_name = 'about';

-- Create a new post if it does not exist.

INSERT INTO wp_posts (post_name, post_type, post_title, post_content)
SELECT 'about', 'page', 'About', 'Like and subscribe'
WHERE NOT EXISTS (
    SELECT 1
    FROM wp_posts
    WHERE post_name = 'about'
);

注意: 为了简洁和清晰起见,此示例省略了许多 WordPress 所需的 wp_posts 列。

了解 INSERT ... SELECT 语句对我来说是“顿悟”时刻。它的目的是使用另一个表的查询结果来一次执行多次插入。但是,您可以通过提供自己的值并仅在 post_name 为“about”的帖子不存在时执行插入来使用和滥用 SQL 的功能。

这是解决这个问题最优雅、最有效的解决方案吗?可能不会。但对于 WordPress 站点的一次性迁移来说,它已经足够好了,并且它允许您更新或插入一行,而无需存储过程或任何应用程序逻辑。希望这篇文章可以帮助您在不借助 ORM 的情况下编写强大的查询。

版本声明 本文转载于:https://dev.to/tylerlwsmith/upsert-a-row-in-a-db-that-doesnt-use-primary-keys-or-unique-constraints-2jnl?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 为什么使用Firefox后退按钮时JavaScript执行停止?
    为什么使用Firefox后退按钮时JavaScript执行停止?
    导航历史记录问题:JavaScript使用Firefox Back Back 此行为是由浏览器缓存JavaScript资源引起的。要解决此问题并确保在后续页面访问中执行脚本,Firefox用户应设置一个空功能。 警报'); }; alert('inline Alert')...
    编程 发布于2025-05-02
  • 为什么不````''{margin:0; }`始终删除CSS中的最高边距?
    为什么不````''{margin:0; }`始终删除CSS中的最高边距?
    在CSS 问题:不正确的代码: 全球范围将所有余量重置为零,如提供的代码所建议的,可能会导致意外的副作用。解决特定的保证金问题是更建议的。 例如,在提供的示例中,将以下代码添加到CSS中,将解决余量问题: body H1 { 保证金顶:-40px; } 此方法更精确,避免了由全局保证金重置引...
    编程 发布于2025-05-02
  • 在PHP中如何高效检测空数组?
    在PHP中如何高效检测空数组?
    在PHP 中检查一个空数组可以通过各种方法在PHP中确定一个空数组。如果需要验证任何数组元素的存在,则PHP的松散键入允许对数组本身进行直接评估:一种更严格的方法涉及使用count()函数: if(count(count($ playerList)=== 0){ //列表为空。 } 对...
    编程 发布于2025-05-02
  • PHP阵列键值异常:了解07和08的好奇情况
    PHP阵列键值异常:了解07和08的好奇情况
    PHP数组键值问题,使用07&08 在给定数月的数组中,键值07和08呈现令人困惑的行为时,就会出现一个不寻常的问题。运行print_r($月份)返回意外结果:键“ 07”丢失,而键“ 08”分配给了9月的值。此问题源于PHP对领先零的解释。当一个数字带有0(例如07或08)的前缀时,PHP将...
    编程 发布于2025-05-02
  • PHP与C++函数重载处理的区别
    PHP与C++函数重载处理的区别
    作为经验丰富的C开发人员脱离谜题,您可能会遇到功能超载的概念。这个概念虽然在C中普遍,但在PHP中构成了独特的挑战。让我们深入研究PHP功能过载的复杂性,并探索其提供的可能性。在PHP中理解php的方法在PHP中,函数超载的概念(如C等语言)不存在。函数签名仅由其名称定义,而与他们的参数列表无关。...
    编程 发布于2025-05-02
  • Android如何向PHP服务器发送POST数据?
    Android如何向PHP服务器发送POST数据?
    在android apache httpclient(已弃用) httpclient httpclient = new defaulthttpclient(); httppost httppost = new httppost(“ http://www.yoursite.com/script.p...
    编程 发布于2025-05-02
  • 如何从PHP中的数组中提取随机元素?
    如何从PHP中的数组中提取随机元素?
    从阵列中的随机选择,可以轻松从数组中获取随机项目。考虑以下数组:; 从此数组中检索一个随机项目,利用array_rand( array_rand()函数从数组返回一个随机键。通过将$项目数组索引使用此键,我们可以从数组中访问一个随机元素。这种方法为选择随机项目提供了一种直接且可靠的方法。
    编程 发布于2025-05-02
  • Go语言垃圾回收如何处理切片内存?
    Go语言垃圾回收如何处理切片内存?
    Garbage Collection in Go Slices: A Detailed AnalysisIn Go, a slice is a dynamic array that references an underlying array.使用切片时,了解垃圾收集行为至关重要,以避免潜在的内存泄...
    编程 发布于2025-05-02
  • \“(1)vs.(;;):编译器优化是否消除了性能差异?\”
    \“(1)vs.(;;):编译器优化是否消除了性能差异?\”
    答案: 在大多数现代编译器中,while(1)和(1)和(;;)之间没有性能差异。编译器: perl: 1 输入 - > 2 2 NextState(Main 2 -E:1)V-> 3 9 Leaveloop VK/2-> A 3 toterloop(next-> 8 last-> 9 ...
    编程 发布于2025-05-02
  • 如何使用替换指令在GO MOD中解析模块路径差异?
    如何使用替换指令在GO MOD中解析模块路径差异?
    在使用GO MOD时,在GO MOD 中克服模块路径差异时,可能会遇到冲突,其中可能会遇到一个冲突,其中3派对软件包将另一个带有导入套件的path package the Imptioned package the Imptioned package the Imported tocted pac...
    编程 发布于2025-05-02
  • 如何在无序集合中为元组实现通用哈希功能?
    如何在无序集合中为元组实现通用哈希功能?
    在未订购的集合中的元素要纠正此问题,一种方法是手动为特定元组类型定义哈希函数,例如: template template template 。 struct std :: hash { size_t operator()(std :: tuple const&tuple)const {...
    编程 发布于2025-05-02
  • 如何将PANDAS DataFrame列转换为DateTime格式并按日期过滤?
    如何将PANDAS DataFrame列转换为DateTime格式并按日期过滤?
    Transform Pandas DataFrame Column to DateTime FormatScenario:Data within a Pandas DataFrame often exists in various formats, including strings.使用时间数据时...
    编程 发布于2025-05-02
  • MySQL中如何高效地根据两个条件INSERT或UPDATE行?
    MySQL中如何高效地根据两个条件INSERT或UPDATE行?
    在两个条件下插入或更新或更新 solution:的答案在于mysql的插入中...在重复键更新语法上。如果不存在匹配行或更新现有行,则此功能强大的功能可以通过插入新行来进行有效的数据操作。如果违反了唯一的密钥约束。实现所需的行为,该表必须具有唯一的键定义(在这种情况下为'名称'...
    编程 发布于2025-05-02
  • 同实例无需转储复制MySQL数据库方法
    同实例无需转储复制MySQL数据库方法
    在同一实例上复制一个MySQL数据库而无需转储在同一mySQL实例上复制数据库,而无需创建InterMediate sqql script。以下方法为传统的转储和IMPORT过程提供了更简单的替代方法。 直接管道数据 MySQL手动概述了一种允许将mysqldump直接输出到MySQL clie...
    编程 发布于2025-05-02
  • 在Python中如何在一个类内从一个函数调用另一个函数?
    在Python中如何在一个类内从一个函数调用另一个函数?
    在类中调用函数:一种实用方法The Issue:Consider the following code that calculates distances between coordinates:class Coordinates: DEF DISTOPONT(self,p): ...
    编程 发布于2025-05-02

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

Copyright© 2022 湘ICP备2022001581号-3