草庐IT

java - 指令重新排序和发生之前的关系

coder 2023-05-13 原文

在 Java Concurrency In Practice 一书中,我们多次被告知,程序的指令可以由编译器、运行时的 JVM 甚至处理器重新排序。所以我们应该假设执行的程序不会按照与我们在源代码中指定的顺序完全相同的顺序执行其指令。

然而,讨论 Java 内存模型的最后一章提供了 happens-before 规则列表,表明 JVM 保留了哪些指令顺序。这些规则中的第一条是:

  • “程序顺序规则。线程中的每个操作都发生在该线程中按程序顺序后面的每个操作之前。”

我相信“程序顺序”是指源代码。

我的问题:假设这条规则,我想知道实际上可能会重新排序什么指令。

“ Action ”定义如下:

The Java Memory Model is specified in terms of actions, which include reads and writes to variables, locks and unlocks of monitors, and starting and joining with threads. The JMM defines a partial ordering called happens before on all actions within the program. To guarantee that the thread executing action B can see the results of action A (whether or not A and B occur in different threads), there must be a happens before relationship between A and B. In the absence of a happens before ordering between two operations, the JVM is free to reorder them as it pleases.

提到的其他订单规则有:

  • 监控锁定规则。监视器锁上的解锁发生在同一监视器锁上的每个后续锁之前。
  • 易变的变量规则。对 volatile 字段的写入发生在对同一字段的每次后续读取之前。
  • 线程启动规则。线程上的 Thread.start 调用发生在已启动线程中的每个操作之前。
  • 线程终止规则。线程中的任何操作发生在任何其他线程检测到该线程已终止之前,通过从 Thread.join 成功返回或通过 Thread.isAlive 返回 false。
  • 中断规则。一个线程在另一个线程上调用中断发生在被中断线程检测到中断之前(通过引发 InterruptedException,或者调用 isInterrupted 或中断)。
  • 终结器规则。对象的构造函数结束发生在该对象的终结器开始之前。
  • 传递性。如果 A 在 B 之前发生,B 在 C 之前发生,那么 A 在 C 之前发生。

最佳答案

程序顺序规则的重点是:在一个线程中

想象一下这个简单的程序(所有变量最初都是 0):

T1:

x = 5;
y = 6;

T2:

if (y == 6) System.out.println(x);

从 T1 的角度来看,执行必​​须与在 x 之后分配的 y(程序顺序)一致。然而,从 T2 的角度来看,情况并非如此,T2 可能会打印 0。

实际上允许 T1 先分配 y,因为这 2 个分配是独立的,交换它们不会影响 T1 的执行。

如果同步正确,T2 将始终打印 5 或什么也不打印。

编辑

您似乎误解了程序顺序的含义。 The program order rule boils down to :

If x and y are actions of the same thread and x comes before y in program order, then hb(x, y) (i.e. x happens-before y).

happens-before 在 JMM 中具有非常特殊的含义。特别是,它意味着从挂钟的角度来看,y=6 必须在 T1 中的 x=5 之后。这只意味着 T1 执行的 Action 顺序必须该顺序一致。也可以引用JLS 17.4.5 :

It should be noted that the presence of a happens-before relationship between two actions does not necessarily imply that they have to take place in that order in an implementation. If the reordering produces results consistent with a legal execution, it is not illegal.

在我上面给出的示例中,您会同意,从 T1 的角度来看(即在单线程程序中),x=5;y=6;y=6 一致;x=5; 因为您没有读取这些值。在 T1 中,下一行的语句保证可以查看这 2 个操作,无论它们执行的顺序如何。

关于java - 指令重新排序和发生之前的关系,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16213443/

有关java - 指令重新排序和发生之前的关系的更多相关文章

  1. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  2. ruby-on-rails - active_admin 目录中的常量警告重新声明 - 2

    我正在使用active_admin,我在Rails3应用程序的应用程序中有一个目录管理,其中包含模型和页面的声明。时不时地我也有一个类,当那个类有一个常量时,就像这样:classFooBAR="bar"end然后,我在每个必须在我的Rails应用程序中重新加载一些代码的请求中收到此警告:/Users/pupeno/helloworld/app/admin/billing.rb:12:warning:alreadyinitializedconstantBAR知道发生了什么以及如何避免这些警告吗? 最佳答案 在纯Ruby中:classA

  3. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  4. ruby - 如何在 Rails 4 中使用表单对象之前的验证回调? - 2

    我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务Controller#创建Action:defcreate@service=Service.new@service_form=ServiceFormObject.new(@service)@service_form.validate(params[:service_form_object])and@service_form.saverespond_with(@service_form,location:admin_services_path)end在验证@ser

  5. ruby - 在 Ruby 中重新分配常量时抛出异常? - 2

    我早就知道Ruby中的“常量”(即大写的变量名)不是真正常量。与其他编程语言一样,对对象的引用是唯一存储在变量/常量中的东西。(侧边栏:Ruby确实具有“卡住”引用对象不被修改的功能,据我所知,许多其他语言都没有提供这种功能。)所以这是我的问题:当您将一个值重新分配给常量时,您会收到如下警告:>>FOO='bar'=>"bar">>FOO='baz'(irb):2:warning:alreadyinitializedconstantFOO=>"baz"有没有办法强制Ruby抛出异常而不是打印警告?很难弄清楚为什么有时会发生重新分配。 最佳答案

  6. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  7. ruby - Rails 关联 - 同一个类的多个 has_one 关系 - 2

    我的问题的一个例子是体育游戏。一场体育比赛有两支球队,一支主队和一支客队。我的事件记录模型如下:classTeam"Team"has_one:away_team,:class_name=>"Team"end我希望能够通过游戏访问一个团队,例如:Game.find(1).home_team但我收到一个单元化常量错误:Game::team。谁能告诉我我做错了什么?谢谢, 最佳答案 如果Gamehas_one:team那么Rails假设您的teams表有一个game_id列。不过,您想要的是games表有一个team_id列,在这种情况下

  8. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

  9. ruby-on-rails - 启用 Rack::Deflater 时 ETag 发生变化 - 2

    在启用Rack::Deflater来gzip我的响应主体时偶然发现了一些奇怪的东西。也许我遗漏了一些东西,但启用此功能后,响应被压缩,但是资源的ETag在每个请求上都会发生变化。这会强制应用程序每次都响应,而不是发送304。这在没有启用Rack::Deflater的情况下有效,我已经验证页面源没有改变。我正在运行一个使用thin作为Web服务器的Rails应用程序。Gemfile.lockhttps://gist.github.com/2510816有没有什么方法可以让我从Rack中间件获得更多的输出,这样我就可以看到发生了什么?提前致谢。 最佳答案

  10. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

随机推荐