java多线程总结四:volatile、synchronized示例

1、synchronized保证同步

先看一个生成偶数的类

[java] view plaincopy

  1. <span style="font-size:16px;">package demo.thread;
  2. /**
  3. *这是一个int生成器的抽象类
  4. *
  5. */
  6. public abstract class IntGenerator {
  7. private volatile boolean canceled = false;
  8. public abstract int next();
  9. public void cancel() {
  10. canceled = true;
  11. }
  12. public boolean isCanceled() {
  13. return canceled;
  14. }
  15. }
  16. </span>

[java] view plaincopy

  1. <span style="font-size:16px;">/*
  2. * 产生偶数
  3. */
  4. class EvenGenerator extends IntGenerator {
  5. private int currentEvenValue = 0;
  6. String s = "";
  7. @Override
  8. public int next() {
  9. <span style="color:#ff0000;">synchronized </span>(s) {
  10. ++currentEvenValue;
  11. ++currentEvenValue;
  12. return currentEvenValue;
  13. }
  14. }
  15. //  //这样也可以
  16. //  public <span style="color:#ff0000;">synchronized </span>int next() {
  17. //          ++currentEvenValue;
  18. //          ++currentEvenValue;
  19. //          return currentEvenValue;
  20. //  }
  21. }</span>

注意到在产生偶数是要加同步锁,否则可能线程1刚好执行了一句++currentEvenValue;操作,就被线程2抢去了cpu,此时线程2执行return currentEvenValue;这时返回的就是一个奇数。加synchronized
就是两个线程同时只能一个线程执行synchronized 块的代码。

测试代码:

[java] view plaincopy

  1. <span style="font-size:16px;">package demo.thread;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. /*
  5. * 消费数字
  6. */
  7. public class EvenChecker implements Runnable {
  8. private IntGenerator generator;
  9. private final int id;
  10. public EvenChecker(IntGenerator g, int ident) {
  11. generator = g;
  12. id = ident;
  13. }
  14. public void run() {
  15. while (!generator.isCanceled()) {
  16. int val = generator.next();
  17. if (val % 2 != 0) {//如果不是偶数
  18. System.out.println(val + " not enen!");
  19. generator.cancel();
  20. }
  21. }
  22. }
  23. public static void test(IntGenerator gp, int count) {
  24. ExecutorService exec = Executors.newCachedThreadPool();
  25. for (int i = 0; i < count; i++)
  26. exec.execute(new EvenChecker(gp, i));
  27. exec.shutdown();
  28. }
  29. public static void test(IntGenerator gp) {
  30. test(gp, 10);
  31. }
  32. public static void main(String[] args) {
  33. test(new EvenGenerator());
  34. }
  35. }</span>

分析:如果产生偶数的类未加synchronized,那么测试程序将会出现奇数导致退出程序。

2、volatile表示原子性,可见性。

对于多个线程之间共享的变量,每个线程都有自己的一份拷贝,当线程1改变变量值时,其他线程并不马上知道该变量值改变了,volatile就保证了变量值对各个线程可见,一个线程改变该值,马上其他线程中该值也改变。原子性表明操作不可中断,如基本变量赋值。

代码示例:

[java] view plaincopy

  1. <span style="font-size:16px;">package demo.thread;
  2. public class VolatileDemo implements Runnable {
  3. private volatile int i = 0;//volatile设置可见性
  4. public synchronized  int getValue() {
  5. return i;
  6. }
  7. private synchronized void enenIncrement() {
  8. i++;
  9. i++;
  10. }
  11. @Override
  12. public void run() {
  13. while (true)
  14. enenIncrement();
  15. }
  16. public static void main(String[] args) {
  17. VolatileDemo at = new VolatileDemo();
  18. new Thread(at).start();
  19. while (true) {
  20. int val = at.getValue();
  21. if (val % 2 != 0) {//出现奇数,退出程序
  22. System.out.println(val+" is not enen!");
  23. System.exit(0);
  24. }
  25. }
  26. }
  27. }
  28. </span>

注意i++操作并不是原子行操作,getValue() 方法也要加synchronized 。

时间: 2024-08-20 05:13:40

java多线程总结四:volatile、synchronized示例的相关文章

Java多线程(四)—— synchronized关键字续

1.synchronized原理 在java中,每一个对象有且仅有一个同步锁.这也意味着,同步锁是依赖于对象而存在.当我们调用某对象的synchronized方法时,就获取了该对象的同步锁.例如,synchronized(obj)就获取了“obj这个对象”的同步锁.不同线程对同步锁的访问是互斥的.也就是说,某时间点,对象的同步锁只能被一个线程获取到!通过同步锁,我们就能在多线程中,实现对“对象/方法”的互斥访问. 例如,现在有两个线程A和线程B,它们都会访问“对象obj的同步锁”.假设,在某一时

Java 多线程(六) synchronized关键字详解

Java 多线程(六) synchronized关键字详解 多线程的同步机制对资源进行加锁,使得在同一个时间,只有一个线程可以进行操作,同步用以解决多个线程同时访问时可能出现的问题. 同步机制可以使用synchronized关键字实现. 当synchronized关键字修饰一个方法的时候,该方法叫做同步方法. 当synchronized方法执行完或发生异常时,会自动释放锁. 下面通过一个例子来对synchronized关键字的用法进行解析. 1.是否使用synchronized关键字的不同 例子

Java多线程(四)、线程池(转)

Java多线程(四).线程池 分类: javaSE综合知识点 2012-09-19 17:46 3943人阅读 评论(1) 收藏 举报 系统启动一个新线程的成本是比较高的,因为它涉及到与操作系统的交互.在这种情况下,使用线程池可以很好的提供性能,尤其是当程序中需要创建大量生存期很短暂的线程时,更应该考虑使用线程池. 与数据库连接池类似的是,线程池在系统启动时即创建大量空闲的线程,程序将一个Runnable对象传给线程池,线程池就会启动一条线程来执行该对象的run方法,当run方法执行结束后,该线

“全栈2019”Java多线程第四章:设置和获取线程名称

难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多线程第四章:设置和获取线程名称 下一章 "全栈2019"Java多线程第五章:线程睡眠sleep()方法详解 学习小组 加入同步学习小组,共同交流与进步. 方式一:关注头条号Gorhaf,私信"Java学习小组". 方式二:关注公众号Gorhaf,回复"Jav

“全栈2019”Java多线程第四十二章:获取线程与读写锁的保持数

难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多线程第四十二章:获取线程与读写锁的保持数 下一章 "全栈2019"Java多线程第四十三章:查询是否有线程在等待读写锁 学习小组 加入同步学习小组,共同交流与进步. 方式一:关注头条号Gorhaf,私信"Java学习小组". 方式二:关注公众号Gorhaf,回复&quo

“全栈2019”Java多线程第四十六章:判断任意线程是否已持有写锁

难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多线程第四十六章:判断任意线程是否已持有写锁 下一章 "全栈2019"Java多线程第四十七章:判断锁是否为公平锁isFair() 学习小组 加入同步学习小组,共同交流与进步. 方式一:加入编程圈子. 方式二:关注头条号Gorhaf,私信"Java学习小组". 方式三:关

“全栈2019”Java多线程第四十七章:判断锁是否为公平锁isFair()

难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多线程第四十七章:判断锁是否为公平锁isFair() 下一章 "全栈2019"Java多线程第四十八章:读写锁实战高并发容器 学习小组 加入同步学习小组,共同交流与进步. 方式一:加入编程圈子. 方式二:关注头条号Gorhaf,私信"Java学习小组". 方式三:关注公众

java多线程之内存可见性-synchronized、volatile

1.JMM:Java Memory Model(Java内存模型) 关于synchronized的两条规定: 1.线程解锁前,必须把共享变量的最新值刷新到主内存中 2.线程加锁时,将清空工作内存中共享变量的值,从而使用共享变量时需要从主内存中重新读取最新的值(注意:加锁和解锁需要是同一把锁) 注:线程解锁前对共享变量的修改在下次加锁时对其他线程可见 2.线程执行互斥代码的过程: 1.获得互斥锁 2.清空工作内存 3.从主内存拷贝变量的最新副本到工作内存 4.执行代码 5.将更改后的共享变量的值刷

java多线程中的volatile和synchronized

package com.chzhao; public class Volatiletest extends Thread { private static int count = 0; public void run() { count++; } public static void main(String[] args) { Thread threads[] = new Thread[10000]; for (int i = 0; i < threads.length; i++) { thre

Java多线程系列八——volatile和ThreadLocal

参考资料: http://ifeve.com/java-memory-model-4/ http://www.infoq.com/cn/articles/java-memory-model-1 http://wuchong.me/blog/2014/08/28/how-to-correctly-write-singleton-pattern/ https://en.wikipedia.org/wiki/Singleton_pattern#Java_5_solution https://www.i