出于教育目的,我正在编写一个简单版本的 AtomicLong,其中一个内部变量由 ReentrantReadWriteLock 保护。
这是一个简化的示例:
public class PlainSimpleAtomicLong {
private long value;
private final ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();
public PlainSimpleAtomicLong(long initialValue) {
this.value = initialValue;
}
public long get() {
long result;
rwLock.readLock().lock();
result = value;
rwLock.readLock().unlock();
return result;
}
// incrementAndGet, decrementAndGet, etc. are guarded by rwLock.writeLock()
}
我的问题:由于“值”是非 volatile 的,其他线程是否有可能通过 PlainSimpleAtomicLong.get() 观察到不正确的初始值?
例如。线程 T1 创建 L = new PlainSimpleAtomicLong(42) 并与线程 T2 共享引用。 T2 是否保证将 L.get() 观察为 42?
如果不是,将 this.value = initialValue; 包装到写锁定/解锁中会有所不同吗?
最佳答案
第 17 章从 happens before 关系的角度解释并发代码。在您的示例中,如果您采用两个随机线程,则 this.value = initialValue; 和 result = value; 之间没有 happens-before 关系。
所以如果你有这样的东西:
T1.start();T2.start();...L = new PlainSimpleAtomicLong(42);long value = L.get();唯一的happens-before (hb) 关系(除了每个线程中的程序顺序)是:1 & 2 hb 3,4,5。
但是 4 和 5 没有排序。但是,如果 T1 在 T2 调用 L.get() 之前调用了 L.get()(从挂钟的角度来看),那么您将拥有一个 hb T1 中的unlock() 和T2 中的lock() 之间的关系。
如前所述,我认为您提议的代码不会破坏 JVM/硬件的任何组合,但它可能会破坏 JMM 的理论实现。
至于您建议将构造函数包装在锁定/解锁中,我认为这还不够,因为至少在理论上,T1 可以在运行主体之前释放对 L 的有效引用(非空)构造函数。因此,风险在于 T2 可以在 T1 在构造函数中获取锁之前获取锁。同样,这是一种交错,在当前的 JVM/硬件上可能是不可能的。
总而言之,如果你想要理论上的线程安全,我认为你不能没有 volatile long value,这就是 AtomicLong 的实现方式。 volatile 将保证在发布对象之前初始化该字段。最后请注意,我在这里提到的问题不是由于您的对象不安全(请参阅@BrettOkken 回答),而是基于对象未安全地跨线程发布的情况。
关于java - 用锁保护非 volatile 字段的初始化?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23984043/
在我的gem中,我需要yaml并且在我的本地计算机上运行良好。但是在将我的gem推送到rubygems.org之后,当我尝试使用我的gem时,我收到一条错误消息=>"uninitializedconstantPsych::Syck(NameError)"谁能帮我解决这个问题?附言RubyVersion=>ruby1.9.2,GemVersion=>1.6.2,Bundlerversion=>1.0.15 最佳答案 经过几个小时的研究,我发现=>“YAML使用未维护的Syck库,而Psych使用现代的LibYAML”因此,为了解决
我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss
我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢
我在Rails工作并有以下类(class):classPlayer当我运行时bundleexecrailsconsole然后尝试:a=Player.new("me",5.0,"UCLA")我回来了:=>#我不知道为什么Player对象不会在这里初始化。关于可能导致此问题的操作/解释的任何建议?谢谢,马里奥格 最佳答案 havenoideawhythePlayerobjectwouldn'tbeinitializedhere它没有初始化很简单,因为你还没有初始化它!您已经覆盖了ActiveRecord::Base初始化方法,但您没有调
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
我知道我可以指定某些字段来使用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
我有用于控制用户任务的Rails5API项目,我有以下错误,但并非总是针对相同的Controller和路由。ActionController::RoutingError:uninitializedconstantApi::V1::ApiController我向您描述了一些我的项目,以更详细地解释错误。应用结构路线scopemodule:'api'donamespace:v1do#=>Loginroutesscopemodule:'login'domatch'login',to:'sessions#login',as:'login',via::postend#=>Teamroutessc
我正在阅读一本关于Ruby的书,作者在编写类初始化定义时使用的形式与他在本书前几节中使用的形式略有不同。它看起来像这样:classTicketattr_accessor:venue,:datedefinitialize(venue,date)self.venue=venueself.date=dateendend在本书的前几节中,它的定义如下:classTicketattr_accessor:venue,:datedefinitialize(venue,date)@venue=venue@date=dateendend在第一个示例中使用setter方法与在第二个示例中使用实例变量之间是
我正在尝试使用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
我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我