草庐IT

mysql - SQL update 在update运行的过程中会影响它的子查询吗?

coder 2023-10-02 原文

我只是在编写一个复杂的更新查询,它看起来或多或少像这样:

update table join
    (select y, min(x) as MinX 
     from table
     group by y) as t1
    using (y)
set x = x - MinX

这意味着变量 x 是根据子查询更新的,子查询也处理变量 x - 但不能这个 x 已经被正在运行的更新命令修改了?这不是问题吗?我的意思是,在正常的编程中,您通常必须明确地处理这个问题,即将新值从旧值存储到其他地方,并在工作完成后,用新值替换旧值......但是 SQL 数据库将如何这样做?

我对单个观察或实验不感兴趣。我想从文档或 sql 标准中获取一个片段,说明在这种情况下定义的行为是什么。我使用的是 MySQL,但对其他 PostgresQL、Oracle 等也有效的答案,尤其是一般的 SQL 标准,我们表示赞赏。谢谢!

最佳答案

** 已编辑 **

从目标表中选择

来自 13.2.9.8. Subqueries in the FROM Clause :

Subqueries in the FROM clause can return a scalar, column, row, or table. Subqueries in the FROM clause cannot be correlated subqueries, unless used within the ON clause of a JOIN operation.

所以,是的,您可以执行上述查询。

问题

这里确实有两个问题。存在并发性,或者确保没有其他人从我们脚下更改数据。这是通过锁定处理的。处理新值与旧值的实际修改是使用派生表处理的。

锁定

在上面的查询中,使用 InnoDB,MySQL 首先执行 SELECT,然后分别获取表中每一行的读(共享)锁。如果您在 SELECT 语句中有一个 WHERE 子句,那么只有您选择的记录会被锁定,其中范围也会导致任何间隙被锁定。

A read lock防止任何其他查询获取写锁,因此在读取锁定时无法从其他地方更新记录。

然后,MySQL 分别获取表中每条记录的写(独占)锁。如果您的 UPDATE 语句中有一个 WHERE 子句,那么只有特定的记录会被写入锁定,同样,如果 WHERE 子句选择了一个范围,那么您将锁定一个范围。

任何从之前的 SELECT 中获得读锁的记录都会自动升级为写锁。

A write lock防止其他查询获得读锁或写锁。

您可以使用 Innotop要通过在锁定模式下运行它来查看这一点,启动事务,执行查询(但不要提交),您将在 Innotop 中看到锁。此外,您可以在没有 Innotop 的情况下使用 SHOW ENGINE INNODB STATUS 查看详细信息。

死锁

如果同时运行两个实例,您的查询很容易出现死锁。如果查询 A 获得读锁,然后查询 B 获得读锁,查询 A 必须等待查询 B 的读锁释放才能获得写锁。但是,查询 B 在完成之前不会释放读锁,除非它可以获取写锁,否则它不会完成。查询 A 和查询 B 陷入僵局,因此陷入僵局。

因此,您可能希望执行显式表锁定,既可以避免大量的记录锁定(占用内存并影响性能),也可以避免死锁。

另一种方法是在内部 SELECT 上使用 SELECT ... FOR UPDATE。这从所有行的写锁开始,而不是从读取开始并升级它们。

派生表

对于内部 SELECT,MySQL 创建一个 derived temporary table .派生表是 MySQL 自动创建的临时表中数据的实际非索引副本(与您显式创建并可以添加索引的临时表相对)。

由于 MySQL 使用派生表,因此这是您在问题中引用的临时旧值。换句话说,这里没有魔法。 MySQL 使用临时值,就像您在其他任何地方执行它一样。

您可以通过对 UPDATE 语句执行 EXPLAIN 来查看派生表(MySQL 5.6+ 支持)。

关于mysql - SQL update 在update运行的过程中会影响它的子查询吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10091788/

有关mysql - SQL update 在update运行的过程中会影响它的子查询吗?的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  3. ruby - ECONNRESET (Whois::ConnectionError) - 尝试在 Ruby 中查询 Whois 时出错 - 2

    我正在用Ruby编写一个简单的程序来检查域列表是否被占用。基本上它循环遍历列表,并使用以下函数进行检查。require'rubygems'require'whois'defcheck_domain(domain)c=Whois::Client.newc.query("google.com").available?end程序不断出错(即使我在google.com中进行硬编码),并打印以下消息。鉴于该程序非常简单,我已经没有什么想法了-有什么建议吗?/Library/Ruby/Gems/1.8/gems/whois-2.0.2/lib/whois/server/adapters/base.

  4. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  5. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  6. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  7. ruby - 多个属性的 update_column 方法 - 2

    我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2

  8. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  9. ruby-on-rails - 在 Rails 和 ActiveRecord 中查询时忽略某些字段 - 2

    我知道我可以指定某些字段来使用pluck查询数据库。ids=Item.where('due_at但是我想知道,是否有一种方法可以指定我想避免从数据库查询的某些字段。某种反拔?posts=Post.where(published:true).do_not_lookup(:enormous_field) 最佳答案 Model#attribute_names应该返回列/属性数组。您可以排除其中一些并传递给pluck或select方法。像这样:posts=Post.where(published:true).select(Post.attr

  10. ruby-on-rails - 无法让 rspec、spork 和调试器正常运行 - 2

    GivenIamadumbprogrammerandIamusingrspecandIamusingsporkandIwanttodebug...mmm...let'ssaaay,aspecforPhone.那么,我应该把“require'ruby-debug'”行放在哪里,以便在phone_spec.rb的特定点停止处理?(我所要求的只是一个大而粗的箭头,即使是一个有挑战性的程序员也能看到:-3)我已经尝试了很多位置,除非我没有正确测试它们,否则会发生一些奇怪的事情:在spec_helper.rb中的以下位置:require'rubygems'require'spork'

随机推荐