1 /* 2 * Written by Doug Lea with assistance from members of JCP JSR-166 3 * Expert Group and released to the public domain, as explained at 4 * http://creativecommons.org/publicdomain/zero/1.0/ 5 */ 6 7 package jsr166; 8 9 import java.util.concurrent.Executors; 10 import java.util.concurrent.ExecutorService; 11 import java.util.concurrent.Phaser; 12 import java.util.concurrent.atomic.DoubleAccumulator; 13 14 import junit.framework.Test; 15 import junit.framework.TestSuite; 16 17 public class DoubleAccumulatorTest extends JSR166TestCase { 18 // android-note: Removed because the CTS runner does a bad job of 19 // retrying tests that have suite() declarations. 20 // 21 // public static void main(String[] args) { 22 // main(suite(), args); 23 // } 24 // public static Test suite() { 25 // return new TestSuite(DoubleAccumulatorTest.class); 26 // } 27 28 /** 29 * default constructed initializes to zero 30 */ 31 public void testConstructor() { 32 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 33 assertEquals(0.0, ai.get()); 34 } 35 36 /** 37 * accumulate accumulates given value to current, and get returns current value 38 */ 39 public void testAccumulateAndGet() { 40 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 41 ai.accumulate(2.0); 42 assertEquals(2.0, ai.get()); 43 ai.accumulate(-4.0); 44 assertEquals(2.0, ai.get()); 45 ai.accumulate(4.0); 46 assertEquals(4.0, ai.get()); 47 } 48 49 /** 50 * reset() causes subsequent get() to return zero 51 */ 52 public void testReset() { 53 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 54 ai.accumulate(2.0); 55 assertEquals(2.0, ai.get()); 56 ai.reset(); 57 assertEquals(0.0, ai.get()); 58 } 59 60 /** 61 * getThenReset() returns current value; subsequent get() returns zero 62 */ 63 public void testGetThenReset() { 64 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 65 ai.accumulate(2.0); 66 assertEquals(2.0, ai.get()); 67 assertEquals(2.0, ai.getThenReset()); 68 assertEquals(0.0, ai.get()); 69 } 70 71 /** 72 * toString returns current value. 73 */ 74 public void testToString() { 75 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 76 assertEquals("0.0", ai.toString()); 77 ai.accumulate(1.0); 78 assertEquals(Double.toString(1.0), ai.toString()); 79 } 80 81 /** 82 * intValue returns current value. 83 */ 84 public void testIntValue() { 85 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 86 assertEquals(0, ai.intValue()); 87 ai.accumulate(1.0); 88 assertEquals(1, ai.intValue()); 89 } 90 91 /** 92 * longValue returns current value. 93 */ 94 public void testLongValue() { 95 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 96 assertEquals(0, ai.longValue()); 97 ai.accumulate(1.0); 98 assertEquals(1, ai.longValue()); 99 } 100 101 /** 102 * floatValue returns current value. 103 */ 104 public void testFloatValue() { 105 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 106 assertEquals(0.0f, ai.floatValue()); 107 ai.accumulate(1.0); 108 assertEquals(1.0f, ai.floatValue()); 109 } 110 111 /** 112 * doubleValue returns current value. 113 */ 114 public void testDoubleValue() { 115 DoubleAccumulator ai = new DoubleAccumulator(Double::max, 0.0); 116 assertEquals(0.0, ai.doubleValue()); 117 ai.accumulate(1.0); 118 assertEquals(1.0, ai.doubleValue()); 119 } 120 121 /** 122 * accumulates by multiple threads produce correct result 123 */ 124 public void testAccumulateAndGetMT() { 125 final int incs = 1000000; 126 final int nthreads = 4; 127 final ExecutorService pool = Executors.newCachedThreadPool(); 128 DoubleAccumulator a = new DoubleAccumulator(Double::max, 0.0); 129 Phaser phaser = new Phaser(nthreads + 1); 130 for (int i = 0; i < nthreads; ++i) 131 pool.execute(new AccTask(a, phaser, incs)); 132 phaser.arriveAndAwaitAdvance(); 133 phaser.arriveAndAwaitAdvance(); 134 double expected = incs - 1; 135 double result = a.get(); 136 assertEquals(expected, result); 137 pool.shutdown(); 138 } 139 140 static final class AccTask implements Runnable { 141 final DoubleAccumulator acc; 142 final Phaser phaser; 143 final int incs; 144 volatile double result; 145 AccTask(DoubleAccumulator acc, Phaser phaser, int incs) { 146 this.acc = acc; 147 this.phaser = phaser; 148 this.incs = incs; 149 } 150 151 public void run() { 152 phaser.arriveAndAwaitAdvance(); 153 DoubleAccumulator a = acc; 154 for (int i = 0; i < incs; ++i) 155 a.accumulate(i); 156 result = a.get(); 157 phaser.arrive(); 158 } 159 } 160 161 } 162