`
kabike
  • 浏览: 611907 次
  • 性别: Icon_minigender_1
  • 来自: 大连
社区版块
存档分类
最新评论

AtomicBoolean运用

阅读更多
首先先看如下例子
 private static class BarWorker implements Runnable {

  private static boolean exists = false;

  private String name;

  public BarWorker(String name) {
   this.name = name;
  }

  public void run() {
   if (!exists) {
    exists = true;
    System.out.println(name + " enter");
    System.out.println(name + " working");
    System.out.println(name + " leave");
    exists = false;
   } else {
    System.out.println(name + " give up");
   }
  }

 }

static变量exists用来实现同一时间只有一个worker在工作. 但是假设exists的判断和exists = true;之间有了
其他指令呢
 private static class BarWorker implements Runnable {

  private static boolean exists = false;

  private String name;

  public BarWorker(String name) {
   this.name = name;
  }

  public void run() {
   if (!exists) {
    try {
     TimeUnit.SECONDS.sleep(1);
    } catch (InterruptedException e1) {
     // do nothing
    }
    exists = true;
    System.out.println(name + " enter");
    try {
     System.out.println(name + " working");
     TimeUnit.SECONDS.sleep(2);
    } catch (InterruptedException e) {
     // do nothing
    }
    System.out.println(name + " leave");
    exists = false;
   } else {
    System.out.println(name + " give up");
   }
  }

 }

这时输出是
bar2 enter
bar2 working
bar1 enter
bar1 working
bar1 leave
bar2 leave
看到两个线程同时工作了.
这时可以用AtomicBoolean
private static class BarWorker implements Runnable {

  private static AtomicBoolean exists = new AtomicBoolean(false);

  private String name;

  public BarWorker(String name) {
   this.name = name;
  }

  public void run() {
   if (exists.compareAndSet(false, true)) {
    System.out.println(name + " enter");
    try {
     System.out.println(name + " working");
     TimeUnit.SECONDS.sleep(2);
    } catch (InterruptedException e) {
     // do nothing
    }
    System.out.println(name + " leave");
    exists.set(false);
   }else{
    System.out.println(name + " give up");
   }
  }

 }

因为它提供了原子性操作,其中exists.compareAndSet(false, true)这个操作把比较和赋值操作组成了一个原子操作,
中间不会提供可乘之机.输出为
bar1 enter
bar1 working
bar2 give up
分享到:
评论
5 楼 kabike 2015-08-28  
QQ1289714862 写道
这代码有问题吧 bar1 bar2 那来的 你是不是new了两个Runnable

应该是,太久了已经忘了
4 楼 QQ1289714862 2015-08-26  
这代码有问题吧 bar1 bar2 那来的 你是不是new了两个Runnable
3 楼 kabike 2013-09-24  
annybz 写道
我还想再问一共 问题,

private static class BarWorker implements Runnable { 
 
  private static AtomicBoolean exists = new AtomicBoolean(false); 
 
  private String name; 
 
  public BarWorker(String name) { 
   this.name = name; 
  } 
 
  public void run() { 
   if (exists.compareAndSet(false, true)) { 
    System.out.println(name + " enter"); 
    try { 
     System.out.println(name + " working"); 
     TimeUnit.SECONDS.sleep(2); 
    } catch (InterruptedException e) { 
     // do nothing 
    } 
    System.out.println(name + " leave"); 
    exists.set(false); 
   }else{ 
    System.out.println(name + " give up"); 
   } 
  } 
 
}
会不会第一个线程进入了if的方法,然后执行到 exists.set(false)之后,第二个线程才进入,这个时候不就可以是两个线程运行了吗,可是我执行了很多遍,发现始终只有一个线程在运行,您能给我解释下吗?谢谢

这个本来也没能防止两个线程同时运行,只是防止它俩同时进入临界区.exists.set(false)时,线程离开了临界区
2 楼 annybz 2013-09-22  
我还想再问一共 问题,

private static class BarWorker implements Runnable { 
 
  private static AtomicBoolean exists = new AtomicBoolean(false); 
 
  private String name; 
 
  public BarWorker(String name) { 
   this.name = name; 
  } 
 
  public void run() { 
   if (exists.compareAndSet(false, true)) { 
    System.out.println(name + " enter"); 
    try { 
     System.out.println(name + " working"); 
     TimeUnit.SECONDS.sleep(2); 
    } catch (InterruptedException e) { 
     // do nothing 
    } 
    System.out.println(name + " leave"); 
    exists.set(false); 
   }else{ 
    System.out.println(name + " give up"); 
   } 
  } 
 
}
会不会第一个线程进入了if的方法,然后执行到 exists.set(false)之后,第二个线程才进入,这个时候不就可以是两个线程运行了吗,可是我执行了很多遍,发现始终只有一个线程在运行,您能给我解释下吗?谢谢
1 楼 annybz 2013-09-22  
我试了第一个用exists控制只有一个线程运行的,结果是可能会看到一个线程运行,也可能看到2个线程运行的。


public class BarWorker implements Runnable {

private static Boolean exists = false;
private String name;

public BarWorker(String name) {
this.name = name;
}

@Override
public void run() {
System.out.println(name + "---"+exists);
if (!exists) {
exists = true;
System.out.println(name + "enter");
System.out.println(name + "working");
System.out.println(name + "leaving");
exists = false;

} else {
System.out.println(name + "give up");
}

}

public static void main(String[] args) {
BarWorker work1 = new BarWorker("bar1");
BarWorker work2 = new BarWorker("bar2");

Thread t1 = new Thread(work1);
Thread t2 = new Thread(work2);

t1.start();
t2.start();
}

}

相关推荐

    JAVA后端架构师.pdf

    他们能够熟练应对常见的并发编程问题,掌握至少一个常见中间件的源码,能够运用设计模式、OOA/D进行软件设计,并在微服务层面掌握常见微服务组件的操作、原理和源码。 技术要点: 1. 操作系统基础知识:微内核与宏...

    tuixiangzi_java_atomicpw4_

    5. **Java并发与原子操作**:“atomicpw4”标签暗示了项目中可能使用了Java并发工具包(java.util.concurrent.atomic)中的原子类,如AtomicInteger或AtomicBoolean,以保证在多线程环境下对游戏状态的更新是线程...

    并发编程的艺术

    6. **原子变量**:Atomic类提供了一组原子操作,如AtomicInteger、AtomicBoolean等,它们在多线程环境下保证更新操作的原子性,无需使用锁。 7. **并发工具类**:如CountDownLatch、CyclicBarrier、Semaphore等,...

    Java多线程之进阶篇(二).docx

    - `AtomicBoolean`:用于原子操作`boolean`类型的变量。 - `AtomicInteger` 和 `AtomicLong`:分别用于原子操作`int`和`long`类型的变量,提供了原子性的增加、减少、设置等操作。 2. **引用原子类**: - `...

    java_util_concurrent_user_guide并发工具包详解

    - `AtomicInteger`, `AtomicLong`, `AtomicBoolean`: 提供了原子性的整型、长型和布尔型变量操作,避免了显式的同步。 - `AtomicReference`: 支持原子性地更新引用的对象。 6. **ThreadLocal** - `ThreadLocal` ...

    java5 并发包 (concurrent)思维导图

    - `AtomicInteger`、`AtomicLong`、`AtomicBoolean`等:提供原子操作的整型、长整型和布尔型变量,无需显式同步即可保证线程安全。 - `AtomicReference`、`AtomicStampedReference`、`AtomicMarkableReference`:...

    juc详解juc详解juc详解juc详解juc详解juc详解juc详解

    - `AtomicInteger`、`AtomicLong`、`AtomicBoolean`等提供了原子操作,保证了在多线程环境下的数据一致性,避免了`synchronized`的使用。 6. **线程中断与中断响应**: - `Thread.interrupt()`用于设置线程的中断...

    JAVA并发编程实践.rar

    10. **原子操作与Atomic类**:Java提供了一组原子类,如AtomicInteger、AtomicBoolean等,它们保证了基本类型的原子性操作,无需使用synchronized进行同步。 11. **并发编程设计模式**:如生产者-消费者模式、读写...

    火山安卓多线程技术源码.rar

    "火山安卓多线程技术源码"提供了一套关于如何在Android平台上运用多线程的实例代码。 1. **线程基础** - **线程概念**:线程是程序执行的最小单元,每个进程可以有多个线程同时执行。 - **主线程**:Android中的...

    针对于Executor框架,Java API,线程共享数据

    - **AtomicBoolean**:用于boolean类型的原子操作。 - **AtomicInteger**:适用于int类型的原子操作。 - **AtomicLong**:用于long类型的原子操作。 - **AtomicReference**:提供对引用类型的原子操作。 - **...

    线程的几种控制方式以及线程间的几种通信方式

    5. **事件(Event)**:Java的`java.util.concurrent.atomic.AtomicBoolean`可以模拟一个简单的事件,Python的`threading.Event`对象则提供了更完善的事件机制。 6. **管道(Pipe)**:在多进程通信中常见,但在...

    java多线程学习笔记

    - **Atomic包**:提供原子操作类,如AtomicInteger、AtomicBoolean等,保证操作的原子性和可见性。 - **CompletableFuture**:用于异步计算,支持链式调用和组合多个异步任务。 在提供的源代码中,可能会涵盖上述...

    java并发编程实践

    在实际开发过程中,合理运用Java并发编程技术,不仅可以提高程序性能,还能简化复杂度。 ##### 5.1 减少锁的竞争 - 使用局部变量代替共享变量。 - 尽量减少锁的粒度,采用细粒度锁。 ##### 5.2 使用并发容器 - `...

    JUC

    - AtomicInteger,AtomicLong,AtomicBoolean等:提供原子操作的整数、长整型和布尔类型变量,避免了显式同步。 - AtomicReference,AtomicIntegerArray等:原子操作引用和其他数组类型,确保多线程环境下的数据...

    原子库:没有人要求的科学怪人图书馆

    此外,AtomicBoolean、AtomicIntegerArray、AtomicLongArray等类提供了对基本类型数组的原子操作,使数组元素的修改也能在并发环境中保持线程安全。 在深入理解Java原子库时,我们还需要了解其背后的底层机制,如...

    uncheck:不再处理异常

    8. **`Unchecked` 工具类**:在Java中,`java.util.concurrent.atomic.AtomicBoolean` 和其他并发工具类提供了 `uncheck` 方法,它允许将一个抛出检查异常的函数包装成不抛出异常的函数。这种方法主要用于简化多线程...

    non-blocking-algorithms-java:Java非阻塞算法

    其中,`Atomic`类(如`AtomicInteger`、`AtomicBoolean`等)提供了原子操作,使得多个线程可以安全地访问共享变量而无需锁定。 1. **原子类与CAS操作**:原子类的实现基于无锁(Lock-Free)或比较并交换(Compare-...

Global site tag (gtag.js) - Google Analytics