JMM
什么是JMM?
JMM就是Java内存模型的简称,Java在分配内存上模拟了一个类似于物理机的模型。
绝大多数的运算任务都不可能只靠处理器“计算”来完成,处理器至少要和内存交互,这部分I/O操作很难消除,存储设备和处理器的速度差距不是一个数量级,所以计算机系统不得不使用高速缓存来缓冲内存和处理器速度问题。
- 缓存的一致性:多处理器环境中,每个处理器都有自己的高速缓存,但是他们又共享同一份内存,当多个处理器在用同一块主存区域时,会导致各自的缓存不一致。
- 解决方案:各个处理器访问缓存遵循协议,读写的时候根据协议来操作。
-
内存模型:在特定的操作协议下,对特定的内存或高速缓存进行读写访问的过程抽象。(Java虚拟机的内存模型和硬件的内存模型有很高的可比性。)
voltaile关键字
Java中的关键字volatile是java虚拟机提供的最轻量级的线程同步机制,当一个变量被声明为volatile之后,该变量将具备以下两种特性:
- volatile保证变量对所有线程的可见性,即任何一个线程修改了该变量的值之后,新值对于所有其他线程都是可以立即得知。
但是java中的运算并非全部都是原子操作,因此volatile变量的运行在并发下一样是线程不安全的。由于volatile变量只能保证可见性,只有在符合如下两条规则情况才是线程安全的。在一定程度下仍然需要通过synchronized同步关键字或者加锁机制来保证线程安全
- 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);
}
}
}
}
以上例子是否解决了线程安全的问题?
答案是:否