java多线程编程核心技术(学习笔记二 第二节)

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,所以一直就是死循环状态。

java多线程编程核心技术(学习笔记二 第二节)_第1张图片

将private boolean isRunning 改为volatile private boolean isRunning,程序强制的从公共内存中读取变量的值。

java多线程编程核心技术(学习笔记二 第二节)_第2张图片

使用volatile关键字增加了实例变量在多个线程之间的可见性,但volatile关键字最致命的缺点是不支持原子性。

比较volatile和synchronized

1)关键字volatile是线程同步的轻量级实现,性能比synchronized快。volatile只能修饰于变量,而synchronized可以修饰方法以及代码块。

2)多线程访问volatile不会发生阻塞,而synchronized会出现阻塞

3)volatile能够保证数据的可见性,但不能保证原子性;而synchronized可以保证原子性,也可以间接的保证可见性,因为它会将私有内存和公共内存中的数据做同步。

4)关键字volatile解决的是变量在多个线程之间的可见性;而synchronized关键字解决的是多个线程访问资源的同步性。

关键字volatile的主要使用场合是在多个线程中可以感知实例变量被更改了,并且可以获得最新的值使用,也就是用多个线程读取共享变量时可以获取最新值使用。

java多线程编程核心技术(学习笔记二 第二节)_第3张图片java多线程编程核心技术(学习笔记二 第二节)_第4张图片


使用原子类进行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());
	  }
   }
  
}
java多线程编程核心技术(学习笔记二 第二节)_第5张图片

你可能感兴趣的:(java多线程编程核心技术(学习笔记二 第二节))