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 java.util.concurrent; 8 9 /** 10 * A {@code TimeUnit} represents time durations at a given unit of 11 * granularity and provides utility methods to convert across units, 12 * and to perform timing and delay operations in these units. A 13 * {@code TimeUnit} does not maintain time information, but only 14 * helps organize and use time representations that may be maintained 15 * separately across various contexts. A nanosecond is defined as one 16 * thousandth of a microsecond, a microsecond as one thousandth of a 17 * millisecond, a millisecond as one thousandth of a second, a minute 18 * as sixty seconds, an hour as sixty minutes, and a day as twenty four 19 * hours. 20 * 21 * <p>A {@code TimeUnit} is mainly used to inform time-based methods 22 * how a given timing parameter should be interpreted. For example, 23 * the following code will timeout in 50 milliseconds if the {@link 24 * java.util.concurrent.locks.Lock lock} is not available: 25 * 26 * <pre> {@code 27 * Lock lock = ...; 28 * if (lock.tryLock(50L, TimeUnit.MILLISECONDS)) ...}</pre> 29 * 30 * while this code will timeout in 50 seconds: 31 * <pre> {@code 32 * Lock lock = ...; 33 * if (lock.tryLock(50L, TimeUnit.SECONDS)) ...}</pre> 34 * 35 * Note however, that there is no guarantee that a particular timeout 36 * implementation will be able to notice the passage of time at the 37 * same granularity as the given {@code TimeUnit}. 38 * 39 * @since 1.5 40 * @author Doug Lea 41 */ 42 public enum TimeUnit { 43 NANOSECONDS { toNanos(long d)44 public long toNanos(long d) { return d; } toMicros(long d)45 public long toMicros(long d) { return d/(C1/C0); } toMillis(long d)46 public long toMillis(long d) { return d/(C2/C0); } toSeconds(long d)47 public long toSeconds(long d) { return d/(C3/C0); } toMinutes(long d)48 public long toMinutes(long d) { return d/(C4/C0); } toHours(long d)49 public long toHours(long d) { return d/(C5/C0); } toDays(long d)50 public long toDays(long d) { return d/(C6/C0); } convert(long d, TimeUnit u)51 public long convert(long d, TimeUnit u) { return u.toNanos(d); } excessNanos(long d, long m)52 int excessNanos(long d, long m) { return (int)(d - (m*C2)); } 53 }, 54 MICROSECONDS { toNanos(long d)55 public long toNanos(long d) { return x(d, C1/C0, MAX/(C1/C0)); } toMicros(long d)56 public long toMicros(long d) { return d; } toMillis(long d)57 public long toMillis(long d) { return d/(C2/C1); } toSeconds(long d)58 public long toSeconds(long d) { return d/(C3/C1); } toMinutes(long d)59 public long toMinutes(long d) { return d/(C4/C1); } toHours(long d)60 public long toHours(long d) { return d/(C5/C1); } toDays(long d)61 public long toDays(long d) { return d/(C6/C1); } convert(long d, TimeUnit u)62 public long convert(long d, TimeUnit u) { return u.toMicros(d); } excessNanos(long d, long m)63 int excessNanos(long d, long m) { return (int)((d*C1) - (m*C2)); } 64 }, 65 MILLISECONDS { toNanos(long d)66 public long toNanos(long d) { return x(d, C2/C0, MAX/(C2/C0)); } toMicros(long d)67 public long toMicros(long d) { return x(d, C2/C1, MAX/(C2/C1)); } toMillis(long d)68 public long toMillis(long d) { return d; } toSeconds(long d)69 public long toSeconds(long d) { return d/(C3/C2); } toMinutes(long d)70 public long toMinutes(long d) { return d/(C4/C2); } toHours(long d)71 public long toHours(long d) { return d/(C5/C2); } toDays(long d)72 public long toDays(long d) { return d/(C6/C2); } convert(long d, TimeUnit u)73 public long convert(long d, TimeUnit u) { return u.toMillis(d); } excessNanos(long d, long m)74 int excessNanos(long d, long m) { return 0; } 75 }, 76 SECONDS { toNanos(long d)77 public long toNanos(long d) { return x(d, C3/C0, MAX/(C3/C0)); } toMicros(long d)78 public long toMicros(long d) { return x(d, C3/C1, MAX/(C3/C1)); } toMillis(long d)79 public long toMillis(long d) { return x(d, C3/C2, MAX/(C3/C2)); } toSeconds(long d)80 public long toSeconds(long d) { return d; } toMinutes(long d)81 public long toMinutes(long d) { return d/(C4/C3); } toHours(long d)82 public long toHours(long d) { return d/(C5/C3); } toDays(long d)83 public long toDays(long d) { return d/(C6/C3); } convert(long d, TimeUnit u)84 public long convert(long d, TimeUnit u) { return u.toSeconds(d); } excessNanos(long d, long m)85 int excessNanos(long d, long m) { return 0; } 86 }, 87 MINUTES { toNanos(long d)88 public long toNanos(long d) { return x(d, C4/C0, MAX/(C4/C0)); } toMicros(long d)89 public long toMicros(long d) { return x(d, C4/C1, MAX/(C4/C1)); } toMillis(long d)90 public long toMillis(long d) { return x(d, C4/C2, MAX/(C4/C2)); } toSeconds(long d)91 public long toSeconds(long d) { return x(d, C4/C3, MAX/(C4/C3)); } toMinutes(long d)92 public long toMinutes(long d) { return d; } toHours(long d)93 public long toHours(long d) { return d/(C5/C4); } toDays(long d)94 public long toDays(long d) { return d/(C6/C4); } convert(long d, TimeUnit u)95 public long convert(long d, TimeUnit u) { return u.toMinutes(d); } excessNanos(long d, long m)96 int excessNanos(long d, long m) { return 0; } 97 }, 98 HOURS { toNanos(long d)99 public long toNanos(long d) { return x(d, C5/C0, MAX/(C5/C0)); } toMicros(long d)100 public long toMicros(long d) { return x(d, C5/C1, MAX/(C5/C1)); } toMillis(long d)101 public long toMillis(long d) { return x(d, C5/C2, MAX/(C5/C2)); } toSeconds(long d)102 public long toSeconds(long d) { return x(d, C5/C3, MAX/(C5/C3)); } toMinutes(long d)103 public long toMinutes(long d) { return x(d, C5/C4, MAX/(C5/C4)); } toHours(long d)104 public long toHours(long d) { return d; } toDays(long d)105 public long toDays(long d) { return d/(C6/C5); } convert(long d, TimeUnit u)106 public long convert(long d, TimeUnit u) { return u.toHours(d); } excessNanos(long d, long m)107 int excessNanos(long d, long m) { return 0; } 108 }, 109 DAYS { toNanos(long d)110 public long toNanos(long d) { return x(d, C6/C0, MAX/(C6/C0)); } toMicros(long d)111 public long toMicros(long d) { return x(d, C6/C1, MAX/(C6/C1)); } toMillis(long d)112 public long toMillis(long d) { return x(d, C6/C2, MAX/(C6/C2)); } toSeconds(long d)113 public long toSeconds(long d) { return x(d, C6/C3, MAX/(C6/C3)); } toMinutes(long d)114 public long toMinutes(long d) { return x(d, C6/C4, MAX/(C6/C4)); } toHours(long d)115 public long toHours(long d) { return x(d, C6/C5, MAX/(C6/C5)); } toDays(long d)116 public long toDays(long d) { return d; } convert(long d, TimeUnit u)117 public long convert(long d, TimeUnit u) { return u.toDays(d); } excessNanos(long d, long m)118 int excessNanos(long d, long m) { return 0; } 119 }; 120 121 // Handy constants for conversion methods 122 static final long C0 = 1L; 123 static final long C1 = C0 * 1000L; 124 static final long C2 = C1 * 1000L; 125 static final long C3 = C2 * 1000L; 126 static final long C4 = C3 * 60L; 127 static final long C5 = C4 * 60L; 128 static final long C6 = C5 * 24L; 129 130 static final long MAX = Long.MAX_VALUE; 131 132 /** 133 * Scale d by m, checking for overflow. 134 * This has a short name to make above code more readable. 135 */ x(long d, long m, long over)136 static long x(long d, long m, long over) { 137 if (d > over) return Long.MAX_VALUE; 138 if (d < -over) return Long.MIN_VALUE; 139 return d * m; 140 } 141 142 // To maintain full signature compatibility with 1.5, and to improve the 143 // clarity of the generated javadoc (see 6287639: Abstract methods in 144 // enum classes should not be listed as abstract), method convert 145 // etc. are not declared abstract but otherwise act as abstract methods. 146 147 /** 148 * Convert the given time duration in the given unit to this 149 * unit. Conversions from finer to coarser granularities 150 * truncate, so lose precision. For example converting 151 * {@code 999} milliseconds to seconds results in 152 * {@code 0}. Conversions from coarser to finer granularities 153 * with arguments that would numerically overflow saturate to 154 * {@code Long.MIN_VALUE} if negative or {@code Long.MAX_VALUE} 155 * if positive. 156 * 157 * <p>For example, to convert 10 minutes to milliseconds, use: 158 * {@code TimeUnit.MILLISECONDS.convert(10L, TimeUnit.MINUTES)} 159 * 160 * @param sourceDuration the time duration in the given {@code sourceUnit} 161 * @param sourceUnit the unit of the {@code sourceDuration} argument 162 * @return the converted duration in this unit, 163 * or {@code Long.MIN_VALUE} if conversion would negatively 164 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 165 */ convert(long sourceDuration, TimeUnit sourceUnit)166 public long convert(long sourceDuration, TimeUnit sourceUnit) { 167 throw new AbstractMethodError(); 168 } 169 170 /** 171 * Equivalent to {@code NANOSECONDS.convert(duration, this)}. 172 * @param duration the duration 173 * @return the converted duration, 174 * or {@code Long.MIN_VALUE} if conversion would negatively 175 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 176 * @see #convert 177 */ toNanos(long duration)178 public long toNanos(long duration) { 179 throw new AbstractMethodError(); 180 } 181 182 /** 183 * Equivalent to {@code MICROSECONDS.convert(duration, this)}. 184 * @param duration the duration 185 * @return the converted duration, 186 * or {@code Long.MIN_VALUE} if conversion would negatively 187 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 188 * @see #convert 189 */ toMicros(long duration)190 public long toMicros(long duration) { 191 throw new AbstractMethodError(); 192 } 193 194 /** 195 * Equivalent to {@code MILLISECONDS.convert(duration, this)}. 196 * @param duration the duration 197 * @return the converted duration, 198 * or {@code Long.MIN_VALUE} if conversion would negatively 199 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 200 * @see #convert 201 */ toMillis(long duration)202 public long toMillis(long duration) { 203 throw new AbstractMethodError(); 204 } 205 206 /** 207 * Equivalent to {@code SECONDS.convert(duration, this)}. 208 * @param duration the duration 209 * @return the converted duration, 210 * or {@code Long.MIN_VALUE} if conversion would negatively 211 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 212 * @see #convert 213 */ toSeconds(long duration)214 public long toSeconds(long duration) { 215 throw new AbstractMethodError(); 216 } 217 218 /** 219 * Equivalent to {@code MINUTES.convert(duration, this)}. 220 * @param duration the duration 221 * @return the converted duration, 222 * or {@code Long.MIN_VALUE} if conversion would negatively 223 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 224 * @see #convert 225 * @since 1.6 226 */ toMinutes(long duration)227 public long toMinutes(long duration) { 228 throw new AbstractMethodError(); 229 } 230 231 /** 232 * Equivalent to {@code HOURS.convert(duration, this)}. 233 * @param duration the duration 234 * @return the converted duration, 235 * or {@code Long.MIN_VALUE} if conversion would negatively 236 * overflow, or {@code Long.MAX_VALUE} if it would positively overflow. 237 * @see #convert 238 * @since 1.6 239 */ toHours(long duration)240 public long toHours(long duration) { 241 throw new AbstractMethodError(); 242 } 243 244 /** 245 * Equivalent to {@code DAYS.convert(duration, this)}. 246 * @param duration the duration 247 * @return the converted duration 248 * @see #convert 249 * @since 1.6 250 */ toDays(long duration)251 public long toDays(long duration) { 252 throw new AbstractMethodError(); 253 } 254 255 /** 256 * Utility to compute the excess-nanosecond argument to wait, 257 * sleep, join. 258 * @param d the duration 259 * @param m the number of milliseconds 260 * @return the number of nanoseconds 261 */ excessNanos(long d, long m)262 abstract int excessNanos(long d, long m); 263 264 /** 265 * Performs a timed {@link Object#wait(long, int) Object.wait} 266 * using this time unit. 267 * This is a convenience method that converts timeout arguments 268 * into the form required by the {@code Object.wait} method. 269 * 270 * <p>For example, you could implement a blocking {@code poll} 271 * method (see {@link BlockingQueue#poll BlockingQueue.poll}) 272 * using: 273 * 274 * <pre> {@code 275 * public synchronized Object poll(long timeout, TimeUnit unit) 276 * throws InterruptedException { 277 * while (empty) { 278 * unit.timedWait(this, timeout); 279 * ... 280 * } 281 * }}</pre> 282 * 283 * @param obj the object to wait on 284 * @param timeout the maximum time to wait. If less than 285 * or equal to zero, do not wait at all. 286 * @throws InterruptedException if interrupted while waiting 287 */ timedWait(Object obj, long timeout)288 public void timedWait(Object obj, long timeout) 289 throws InterruptedException { 290 if (timeout > 0) { 291 long ms = toMillis(timeout); 292 int ns = excessNanos(timeout, ms); 293 obj.wait(ms, ns); 294 } 295 } 296 297 /** 298 * Performs a timed {@link Thread#join(long, int) Thread.join} 299 * using this time unit. 300 * This is a convenience method that converts time arguments into the 301 * form required by the {@code Thread.join} method. 302 * 303 * @param thread the thread to wait for 304 * @param timeout the maximum time to wait. If less than 305 * or equal to zero, do not wait at all. 306 * @throws InterruptedException if interrupted while waiting 307 */ timedJoin(Thread thread, long timeout)308 public void timedJoin(Thread thread, long timeout) 309 throws InterruptedException { 310 if (timeout > 0) { 311 long ms = toMillis(timeout); 312 int ns = excessNanos(timeout, ms); 313 thread.join(ms, ns); 314 } 315 } 316 317 /** 318 * Performs a {@link Thread#sleep(long, int) Thread.sleep} using 319 * this time unit. 320 * This is a convenience method that converts time arguments into the 321 * form required by the {@code Thread.sleep} method. 322 * 323 * @param timeout the minimum time to sleep. If less than 324 * or equal to zero, do not sleep at all. 325 * @throws InterruptedException if interrupted while sleeping 326 */ sleep(long timeout)327 public void sleep(long timeout) throws InterruptedException { 328 if (timeout > 0) { 329 long ms = toMillis(timeout); 330 int ns = excessNanos(timeout, ms); 331 Thread.sleep(ms, ns); 332 } 333 } 334 335 } 336