2.3 volatile关键字
通过使用volatile关键字,强制的从公共内存中读取变量的值。
package p2;
public class Run6 {
public static void main(String[] args) throws InterruptedException {
RunThread runThread=new RunThread();
runThread.start();
runThread.sleep(1000);
runThread.setRunning(false);
System.out.println("已经赋值为false");
}
}
package p2;
public class RunThread extends Thread{
private boolean isRunning=true;
public boolean isRunning() {
return isRunning;
}
public void setRunning(boolean isRunning) {
this.isRunning=isRunning;
public void run() { System.out.println("进入run了"); while(isRunning=true) { } System.out.println("线程被停止了"); }}
上述代码在jvm设置为-server下会出现死循环,原因是在启动RunThread.java线程时,变量private boolean isRunning=true;存在于公共堆栈及线程的私有堆栈中,在JVM被设置为-server模式时为了线程的运行效率,线程一直在私有堆栈中取得isRunning的值为true,而代码thread.setRunning(false)虽然被执行,更新的缺是公共堆栈中的isRunning变量值为false,所以一直就是死循环状态。
将private boolean isRunning 改为volatile private boolean isRunning,程序强制的从公共内存中读取变量的值。
使用volatile关键字增加了实例变量在多个线程之间的可见性,但volatile关键字最致命的缺点是不支持原子性。
比较volatile和synchronized
1)关键字volatile是线程同步的轻量级实现,性能比synchronized快。volatile只能修饰于变量,而synchronized可以修饰方法以及代码块。
2)多线程访问volatile不会发生阻塞,而synchronized会出现阻塞
3)volatile能够保证数据的可见性,但不能保证原子性;而synchronized可以保证原子性,也可以间接的保证可见性,因为它会将私有内存和公共内存中的数据做同步。
4)关键字volatile解决的是变量在多个线程之间的可见性;而synchronized关键字解决的是多个线程访问资源的同步性。
关键字volatile的主要使用场合是在多个线程中可以感知实例变量被更改了,并且可以获得最新的值使用,也就是用多个线程读取共享变量时可以获取最新值使用。
使用原子类进行i++操作
一个原子类型就是一个原子操作可用的类型,它在没有锁的情况下做到线程安全。
package p2;
public class Run7 {
public static void main(String[] args) throws InterruptedException {
AddCountThread addCountThread=new AddCountThread();
Thread t1=new Thread(addCountThread);
t1.start();
Thread t2=new Thread(addCountThread);
t2.start();
Thread t3=new Thread(addCountThread);
t3.start();
}
}
package p2;
import java.util.concurrent.atomic.AtomicInteger;
public class AddCountThread extends Thread {
private AtomicInteger count=new AtomicInteger(0);
public void run() {
for(int i=0;i<1000;i++) {
System.out.println(count.incrementAndGet());
}
}
}