Java的内存模型是围绕着在并发过程之中如何处理原子性,可见性和有序性这三个特征来建立的。
原子性
由java内存模型来直接保证的原子性变量操作包括read,load,assign,use,store,write,我们大致可以认为基本数据类型的访问读写是具备原子性的(例外就是long和double的非原子协定),如果应用场景需要一个更大范围的原子性保证,java内存模型还提供了lock和unlock的操作来满足这种需求,尽管jvm把lock操作和unlock操作直接开放给用户,但是却提供了更高层次的字节码指令monitorenter
和
monitorexit
来隐式地使用这两个操作,这两个字节码指令反映到java代码之中就是同步块(synchronized
),所以在synchronized
块之间的操作也是具备有原子性的。
可见性
可见性是指当一个线程修改了共享变量的值,其他线程能够立即得知这个修改。java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值这种依赖主内存作为传递媒介的方式来实现可见性的,无论普通变量还是volatile变量都是如此,普通变量与volatile变量的区别在于:volatile的特殊规则保证了新值能立即同步回主内存,以及每次使用前立即从主内存刷新。因此,volatile保证了多线程操作时变量的可见性,而普通变量则不能保证。
除了volatile
,java中还有synchronized
和final
关键字能实现可见性。synchronized
的可见性是由于“对于一个变量执行unlock操作之前,必须先把此变量同步回主内存中
”;而对于final
关键字的可见性来说:被final
关键字修饰的字段在构造器之中一旦初始化完成,并且构造器没有吧this
的引用传递出去,那么其他线程中就能看到final字段的值。
有序性
java程序中天然的有效性可以总结为一句话:如果在本线程内观察,所有的操作都是有序的;(线程内表现为串行的语义)如果在一个线程中观察另外一个线程,所有的操作都是无序的。(指令重排序现象和工作内存与主内存同步延迟现象)。
java提供了volatile
和synchronized
两个关键字来保证线程之间的操作的有序性,volatile
关键字本身就包含了禁止指令重排序的语义,而synchronized
关键字则是由“一个变量在同一时刻只允许一条线程对其进行lock操作”这条规则获取的,这条规则决定了持有同一个锁的两个同步块只能串行地进入。
网友评论