10-java内存模型

JMM

什么是JMM?
JMM就是Java内存模型的简称,Java在分配内存上模拟了一个类似于物理机的模型。

绝大多数的运算任务都不可能只靠处理器“计算”来完成,处理器至少要和内存交互,这部分I/O操作很难消除,存储设备和处理器的速度差距不是一个数量级,所以计算机系统不得不使用高速缓存来缓冲内存和处理器速度问题。

  • 缓存的一致性:多处理器环境中,每个处理器都有自己的高速缓存,但是他们又共享同一份内存,当多个处理器在用同一块主存区域时,会导致各自的缓存不一致。
  • 解决方案:各个处理器访问缓存遵循协议,读写的时候根据协议来操作。
  • 内存模型:在特定的操作协议下,对特定的内存或高速缓存进行读写访问的过程抽象。(Java虚拟机的内存模型和硬件的内存模型有很高的可比性。)


    image

voltaile关键字

Java中的关键字volatile是java虚拟机提供的最轻量级的线程同步机制,当一个变量被声明为volatile之后,该变量将具备以下两种特性:

  1. volatile保证变量对所有线程的可见性,即任何一个线程修改了该变量的值之后,新值对于所有其他线程都是可以立即得知。

但是java中的运算并非全部都是原子操作,因此volatile变量的运行在并发下一样是线程不安全的。由于volatile变量只能保证可见性,只有在符合如下两条规则情况才是线程安全的。在一定程度下仍然需要通过synchronized同步关键字或者加锁机制来保证线程安全

  1. volatile禁止指令重排序优化。 普通变量仅能保证在方法执行过程中所有依赖赋值结果的地方都能获取正确的结果,而无法保证变量赋值操作顺序与程序代码执行顺序一致。volatile禁止指令重排序.

下面是简单的volatile的例子:

public class Main {
    volatile int i=0;
    public static void main(String[] args){
        Thread t1=new Thread(new TheThread());
        Thread t2=new Thread(new TheThread());
        t1.start();
        t2.start();
    }
static class TheThread implements Runnable{ 
    public void run() {
        for(int i=0;i<100;i++){
            System.out.println(i);
        }
    }
}
}

以上例子是否解决了线程安全的问题?
答案是:否

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 目录: 1. 指令重排 2. 顺序一致性 3. volatile 4. final 1.指令重排 要了解指令重排,...
    西部小笼包阅读 784评论 0 1
  • Java的并发采用的是共享内存模型(而非消息传递模型),线程之间共享程序的公共状态,线程之间通过写-读内存中的公共...
    阿斯蒂芬2阅读 502评论 0 1
  • 哇一周咋又过去了呢(ಡωಡ),感觉这一周是我过的最快的一周哈哈哈,因为这一周只要上三天课(๑>؂<๑) 这一...
    Elton2017阅读 242评论 0 2
  • 关于变量和方法的二义性辩论 首先先看个类 两个疑问1、为啥 ‘1’处会有输出,但是‘4’处没有输出?2、为啥 ‘2...
    AQ王浩阅读 504评论 0 2
  • 与往事干杯,今朝不醉 文/斯禅 广州的冬季,终于也来了一场像样的寒冷。每个人都瞬间裹得严严实实的厚,行走在小...
    斯禅阅读 349评论 0 2