JUC线程高级_volatile关键字与内存可见性

volatile关键字与内存可见性

看如下一段代码

package com.atguigu.juc;

/*
 * 一、volatile 关键字:当多个线程进行操作共享数据时,可以保证内存中的数据可见。
 * 					  相较于 synchronized 是一种较为轻量级的同步策略。
 * 
 * 注意:
 * 1. volatile 不具备“互斥性”
 * 2. volatile 不能保证变量的“原子性”
 */
public class TestVolatile {
	
	public static void main(String[] args) {
		ThreadDemo td = new ThreadDemo();
		new Thread(td).start();
		
		while(true){
			if(td.isFlag()){
				System.out.println("------------------");
				break;
			}
		}
		
	}

}

class ThreadDemo implements Runnable {

	private volatile boolean flag = false;

	@Override
	public void run() {
		
		try {
			Thread.sleep(200);
		} catch (InterruptedException e) {
		}

		flag = true;
		
		System.out.println("flag=" + isFlag());

	}

	public boolean isFlag() {
		return flag;
	}

	public void setFlag(boolean flag) {
		this.flag = flag;
	}

}

如果成员变量flag没有用volatile声明,那么打印不出“-----”。
原因是线程1运行了将flag赋值为true,即使main线程和线程td共享数据flag,但是main线程执行速度很快,在td线程将flag=true返回给主存之前,main线程就已经从主存拿到了flag=false。
内存可见性问题就是,当多个线程操作共享数据时,彼此不可见
方法一:
用同步锁解决

		synchronized(td){
		if(td.isFlag()){
				System.out.println("------------------");
				break;
			}
		}

让线程td重复地到主存中去读数据。
但是当多个线程去访问这段时,如果一个线程正在占用这个锁,其他线程就只能挂起,造成阻塞。效率低

volatile关键字
当多个线程去操作共享数据时候,可以保证内存中的数据可见,相较于synchronized是一种轻量级的同步策略

注意:
1.volatile 不具备“互斥性”
2.volatile 不能保证变量的“原子性”。

发布了11 篇原创文章 · 获赞 0 · 访问量 202

猜你喜欢

转载自blog.csdn.net/MuNaiTao/article/details/104642105