java并发编程中实现可见性的四种可行方案解析

 更新时间:2023年08月07日 09:04:53   作者:键盘林  
这篇文章主要介绍了java并发编程中实现可见性的四种可行方案解析,使用关键字volatile和使用锁(如synchronized关键字或者java.util.concurrent包中的锁)来确保对共享变量的修改在多线程环境中能够正确地被其他线程所观察到,需要的朋友可以参考下

在方法或者变量已经同步的情况下,还会出现什么问题吗?

多线程编程主要学的就是"互斥"和"可见","互斥"指的就是就是同步,而"可见"想必很多人还没有理解.

举个例子:

MyVolatile.java:

package cn.mxl.test.wr;
public class MyVolatile {
	private  int a=0;
	public   void write() throws InterruptedException {
		a++;
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
	public   void read() throws InterruptedException {
		while(a==0) {
//			Thread.sleep(1);
//			System.out.println(a);
		}
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
}

MyRead.java:

package cn.mxl.test.wr;
public class MyRead implements Runnable{
	private MyVolatile task;
	public MyRead(MyVolatile task) {
		// TODO Auto-generated constructor stub
		this.task=task;
	}
	@Override
	public void run() {
		// TODO Auto-generated method stub
		try {
			task.read();
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

MyWrite.java:

package cn.mxl.test.wr;
public class MyWrite implements Runnable{
	private MyVolatile task;
	public MyWrite(MyVolatile task) {
		// TODO Auto-generated constructor stub
		this.task=task;
	}
	@Override
	public void run() {
		// TODO Auto-generated method stub
		try {
			task.write();
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

Run.java:

package cn.mxl.test.wr;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.ReentrantLock;
public class Run {
	public static void main(String[] args) throws Exception, ExecutionException {
		MyVolatile task=new MyVolatile();
		MyWrite mt1=new MyWrite(task);
		MyRead mt2=new MyRead(task);
		Thread tW=new Thread(mt1,"Write");
		Thread tR=new Thread(mt2,"Read");
		tR.start();
		Thread.sleep(2000);
		tW.start();
	}
}

运行结果:

Read线程一直处于a==0中是为什么呢?

因为当Read线程 先开始线程的时候,加载主内存中的数据放到自己的私有内存中,在程序执行过程中为了提高效率,它就只读自己私有内存中的数据,不会再去读主内存中的共享数据,在这个过程中,主内存对于该线程来说,只写不读,也就是说将自己修改的变量值存放到主内存中,但是自己不去读主内存的数据;所以说,在Read线程执行到while语句一直循环的时候,即使Write线程修改a==1值了,Read线程还是之前a==0的值,一直跳不出循环;

解决办法-->可见性,让Read线程读a变量的时候去主内存(共享数据)中读取:

volatiel关键字的使用:

MyVolatile.java:

package cn.mxl.test.wr;
public class MyVolatile {
	private volatile  int a=0;
	public   void write() throws InterruptedException {
		a++;
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
	public   void read() throws InterruptedException {
		while(a==0) {
//			Thread.sleep(1);
//			System.out.println(a);
		}
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
}

结果:

其实在这个过程中,我遇到了一些有趣的事情,可见性这东西,其实还有三个东西可以实现:

第一个-->Thread.sleep():

MyVolatile.java(取消sleep的注解):

package cn.mxl.test.wr;
public class MyVolatile {
	private   int a=0;
	public   void write() throws InterruptedException {
		a++;
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
	public   void read() throws InterruptedException {
		while(a==0) {
			Thread.sleep(1);
//			System.out.println(a);
		}
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
}

 结果:

第二个是system.out.println():

MyVolatile.java:

package cn.mxl.test.wr;
public class MyVolatile {
	private   int a=0;
	public   void write() throws InterruptedException {
		a++;
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
	public   void read() throws InterruptedException {
		while(a==0) {
//			Thread.sleep(1);
			System.out.println(a);
		}
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
}

 结果:

 第三个atomic:

MyVolatile.java:

package cn.mxl.test.wr.atomic;
import java.util.concurrent.atomic.AtomicInteger;
public class MyVolatile {
	private   AtomicInteger a=new AtomicInteger();
	public   void write() throws InterruptedException {
		a.incrementAndGet();
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
	public   void read() throws InterruptedException {
		while(a.get()==0) {
//			Thread.sleep(1);
//			System.out.println(a);
		}
		System.out.println("当前线程:"+Thread.currentThread().getName());
		System.out.println(a);
	}
}

结果:

到此这篇关于java并发编程中实现可见性的四种可行方案解析的文章就介绍到这了,更多相关java实现可见性内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java中IO流使用FileWriter写数据基本操作详解

    Java中IO流使用FileWriter写数据基本操作详解

    这篇文章主要介绍了Java中IO流FileWriter写数据操作,FileWriter类提供了多种写入字符的方法,包括写入单个字符、写入字符数组和写入字符串等,它还提供了一些其他的方法,如刷新缓冲区、关闭文件等,需要的朋友可以参考下
    2023-10-10
  • Java远程执行shell命令出现java: command not found问题及解决

    Java远程执行shell命令出现java: command not found问题及解决

    这篇文章主要介绍了Java远程执行shell命令出现java: command not found问题及解决方案,具有很好的参考价值,希望对大家有所帮助。
    2023-07-07
  • Java中关于String StringBuffer StringBuilder特性深度解析

    Java中关于String StringBuffer StringBuilder特性深度解析

    这篇文章主要介绍了Java中关于String StringBuffer StringBuilder特性深度解析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-09-09
  • 详解springboot集成websocket的两种实现方式

    详解springboot集成websocket的两种实现方式

    这篇文章主要介绍了springboot集成websocket的两种实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • 关于Spring Cache 缓存拦截器( CacheInterceptor)

    关于Spring Cache 缓存拦截器( CacheInterceptor)

    这篇文章主要介绍了关于Spring Cache缓存拦截器( CacheInterceptor),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • Java方法引用原理实例解析

    Java方法引用原理实例解析

    这篇文章主要介绍了Java方法引用的原理实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2021-08-08
  • 最新IntelliJ IDEA 2020版本的安装教程详解

    最新IntelliJ IDEA 2020版本的安装教程详解

    这篇文章主要介绍了最新IntelliJ IDEA 2020版本的安装教程详解,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08
  • 浅谈Java中是否直接可以使用enum进行传输

    浅谈Java中是否直接可以使用enum进行传输

    这篇文章主要介绍了浅谈Java中是否直接可以使用enum进行传输,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-05-05
  • Java获取本机IP地址的三种方法总结

    Java获取本机IP地址的三种方法总结

    这篇文章主要为大家详细介绍了java获取本机IP地址的三种方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • maven 打包项目的几种方式

    maven 打包项目的几种方式

    maven目前在web上面的使用方式很普遍,而打包的方式也存在很多方式,本文就详细的介绍了三种方式,具有一定的参考价值,感兴趣的可以了解下
    2021-06-06

最新评论