001/* 002 * Units of Measurement Reference Implementation 003 * Copyright (c) 2005-2018, Jean-Marie Dautelle, Werner Keil, Otavio Santana. 004 * 005 * All rights reserved. 006 * 007 * Redistribution and use in source and binary forms, with or without modification, 008 * are permitted provided that the following conditions are met: 009 * 010 * 1. Redistributions of source code must retain the above copyright notice, 011 * this list of conditions and the following disclaimer. 012 * 013 * 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions 014 * and the following disclaimer in the documentation and/or other materials provided with the distribution. 015 * 016 * 3. Neither the name of JSR-385, Indriya nor the names of their contributors may be used to endorse or promote products 017 * derived from this software without specific prior written permission. 018 * 019 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 020 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, 021 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 022 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 023 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 024 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 025 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 026 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 027 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, 028 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 029 */ 030package tech.units.indriya.unit; 031 032import tech.units.indriya.AbstractSystemOfUnits; 033import tech.units.indriya.AbstractUnit; 034import tech.units.indriya.function.AddConverter; 035import tech.units.indriya.function.RationalConverter; 036import tech.units.indriya.quantity.QuantityDimension; 037import static tech.units.indriya.AbstractUnit.ONE; 038 039import javax.measure.Quantity; 040import javax.measure.Unit; 041import javax.measure.quantity.Acceleration; 042import javax.measure.quantity.AmountOfSubstance; 043import javax.measure.quantity.Angle; 044import javax.measure.quantity.Area; 045import javax.measure.quantity.CatalyticActivity; 046import javax.measure.quantity.Dimensionless; 047import javax.measure.quantity.ElectricCapacitance; 048import javax.measure.quantity.ElectricCharge; 049import javax.measure.quantity.ElectricConductance; 050import javax.measure.quantity.ElectricCurrent; 051import javax.measure.quantity.ElectricInductance; 052import javax.measure.quantity.ElectricPotential; 053import javax.measure.quantity.ElectricResistance; 054import javax.measure.quantity.Energy; 055import javax.measure.quantity.Force; 056import javax.measure.quantity.Frequency; 057import javax.measure.quantity.Illuminance; 058import javax.measure.quantity.Length; 059import javax.measure.quantity.LuminousFlux; 060import javax.measure.quantity.LuminousIntensity; 061import javax.measure.quantity.MagneticFlux; 062import javax.measure.quantity.MagneticFluxDensity; 063import javax.measure.quantity.Mass; 064import javax.measure.quantity.Power; 065import javax.measure.quantity.Pressure; 066import javax.measure.quantity.RadiationDoseAbsorbed; 067import javax.measure.quantity.RadiationDoseEffective; 068import javax.measure.quantity.Radioactivity; 069import javax.measure.quantity.SolidAngle; 070import javax.measure.quantity.Speed; 071import javax.measure.quantity.Temperature; 072import javax.measure.quantity.Time; 073import javax.measure.quantity.Volume; 074import javax.measure.spi.SystemOfUnits; 075 076/** 077 * <p> 078 * This class defines commonly used units. 079 * 080 * @author <a href="mailto:units@catmedia.us">Werner Keil</a> 081 * @version 1.2, June 29, 2018 082 * @since 1.0 083 */ 084public class Units extends AbstractSystemOfUnits { 085 086 protected Units() { 087 } 088 089 private static final Units INSTANCE = new Units(); 090 091 public String getName() { 092 return Units.class.getSimpleName(); 093 } 094 095 // ////////////// 096 // BASE UNITS // 097 // ////////////// 098 099 /** 100 * The SI base unit for electric current quantities (standard name <code>A</code>). The Ampere is that constant current which, if maintained in two 101 * straight parallel conductors of infinite length, of negligible circular cross-section, and placed 1 meter apart in vacuum, would produce between 102 * these conductors a force equal to 2 * 10-7 newton per meter of length. It is named after the French physicist Andre Ampere (1775-1836). 103 * 104 * @implNote SI Base Unit 105 */ 106 public static final Unit<ElectricCurrent> AMPERE = addUnit(new BaseUnit<ElectricCurrent>("A", QuantityDimension.ELECTRIC_CURRENT), 107 ElectricCurrent.class); 108 109 /** 110 * The SI base unit for luminous intensity quantities (standard name <code>cd</code>). The candela is the luminous intensity, in a given direction, 111 * of a source that emits monochromatic radiation of frequency 540 * 1012 hertz and that has a radiant intensity in that direction of 1/683 watt per 112 * steradian 113 * 114 * @see <a href="http://en.wikipedia.org/wiki/Candela"> Wikipedia: Candela</a> 115 * 116 * @implNote SI Base Unit 117 */ 118 public static final Unit<LuminousIntensity> CANDELA = addUnit(new BaseUnit<LuminousIntensity>("cd", QuantityDimension.LUMINOUS_INTENSITY), 119 LuminousIntensity.class); 120 121 /** 122 * The SI base unit for thermodynamic temperature quantities (standard name <code>K</code>). The kelvin is the 1/273.16th of the thermodynamic 123 * temperature of the triple point of water. It is named after the Scottish mathematician and physicist William Thomson 1st Lord Kelvin (1824-1907) 124 * 125 * @implNote SI Base Unit 126 */ 127 public static final Unit<Temperature> KELVIN = addUnit(new BaseUnit<Temperature>("K", QuantityDimension.TEMPERATURE), Temperature.class); 128 129 /** 130 * The SI base unit for mass quantities (standard name <code>kg</code>). It is the only SI unit with a prefix as part of its name and symbol. The 131 * kilogram is equal to the mass of an international prototype in the form of a platinum-iridium cylinder kept at Sevres in France. 132 * 133 * @see #GRAM 134 * 135 * @implNote SI Base Unit 136 */ 137 public static final Unit<Mass> KILOGRAM = addUnit(new BaseUnit<Mass>("kg", QuantityDimension.MASS), Mass.class); 138 139 /** 140 * The SI base unit for length quantities (standard name <code>m</code>). One metre was redefined in 1983 as the distance traveled by light in a 141 * vacuum in 1/299,792,458 of a second. 142 * 143 * @implNote SI Base Unit 144 */ 145 public static final Unit<Length> METRE = addUnit(new BaseUnit<>("m", QuantityDimension.LENGTH), Length.class); 146 147 /** 148 * The SI base unit for amount of substance quantities (standard name <code>mol</code>). The mole is the amount of substance of a system which 149 * contains as many elementary entities as there are atoms in 0.012 kilogram of carbon 12. 150 * 151 * @implNote SI Base Unit 152 */ 153 public static final Unit<AmountOfSubstance> MOLE = addUnit(new BaseUnit<>("mol", QuantityDimension.AMOUNT_OF_SUBSTANCE), AmountOfSubstance.class); 154 155 /** 156 * The SI base unit for duration quantities (standard name <code>s</code>). It is defined as the duration of 9,192,631,770 cycles of radiation 157 * corresponding to the transition between two hyperfine levels of the ground state of cesium (1967 Standard). 158 * 159 * @implNote SI Base Unit 160 */ 161 public static final Unit<Time> SECOND = addUnit(new BaseUnit<>("s", QuantityDimension.TIME), Time.class); 162 163 // ////////////////////////////// 164 // SI DERIVED ALTERNATE UNITS // 165 // ////////////////////////////// 166 167 /** 168 * The SI derived unit for mass quantities (standard name <code>g</code>). The base unit for mass quantity is {@link #KILOGRAM}. 169 */ 170 public static final Unit<Mass> GRAM = addUnit(KILOGRAM.divide(1000)); 171 // = new TransformedUnit(KILOGRAM, MetricPrefix.KILO.getConverter()); 172 173 /** 174 * The SI unit for plane angle quantities (standard name <code>rad</code>). One radian is the angle between two radii of a circle such that the 175 * length of the arc between them is equal to the radius. 176 */ 177 public static final Unit<Angle> RADIAN = addUnit(new AlternateUnit<>(ONE, "rad"), Angle.class); 178 179 /** 180 * The SI unit for solid angle quantities (standard name <code>sr</code>). One steradian is the solid angle subtended at the center of a sphere by 181 * an area on the surface of the sphere that is equal to the radius squared. The total solid angle of a sphere is 4*Pi steradians. 182 */ 183 public static final Unit<SolidAngle> STERADIAN = addUnit(new AlternateUnit<SolidAngle>(ONE, "sr"), SolidAngle.class); 184 185 /** 186 * The SI unit for frequency (standard name <code>Hz</code>). A unit of frequency equal to one cycle per second. After Heinrich Rudolf Hertz 187 * (1857-1894), German physicist who was the first to produce radio waves artificially. 188 */ 189 public static final Unit<Frequency> HERTZ = addUnit(new AlternateUnit<Frequency>(ONE.divide(SECOND), "Hz"), Frequency.class); 190 191 /** 192 * The SI unit for force (standard name <code>N</code>). One newton is the force required to give a mass of 1 kilogram an Force of 1 metre per 193 * second per second. It is named after the English mathematician and physicist Sir Isaac Newton (1642-1727). 194 */ 195 public static final Unit<Force> NEWTON = addUnit(new AlternateUnit<Force>(METRE.multiply(KILOGRAM).divide(SECOND.pow(2)), "N"), Force.class); 196 197 /** 198 * The SI unit for pressure, stress (standard name <code>Pa</code>). One pascal is equal to one newton per square meter. It is named after the 199 * French philosopher and mathematician Blaise Pascal (1623-1662). 200 */ 201 @SuppressWarnings({ "unchecked", "rawtypes" }) 202 public static final Unit<Pressure> PASCAL = addUnit(new AlternateUnit(NEWTON.divide(METRE.pow(2)), "Pa"), Pressure.class); 203 204 /** 205 * The SI unit for energy, work, quantity of heat (<code>J</code>). One joule is the amount of work done when an applied force of 1 newton moves 206 * through a distance of 1 metre in the direction of the force. It is named after the English physicist James Prescott Joule (1818-1889). 207 */ 208 public static final Unit<Energy> JOULE = addUnit(new AlternateUnit<Energy>(NEWTON.multiply(METRE), "J"), Energy.class); 209 210 /** 211 * The SI unit for power, radiant, flux (standard name <code>W</code>). One watt is equal to one joule per second. It is named after the British 212 * scientist James Watt (1736-1819). 213 */ 214 public static final Unit<Power> WATT = addUnit(new AlternateUnit<Power>(JOULE.divide(SECOND), "W"), Power.class); 215 216 /** 217 * The SI unit for electric charge, quantity of electricity (standard name <code>C</code>). One Coulomb is equal to the quantity of charge 218 * transferred in one second by a steady current of one ampere. It is named after the French physicist Charles Augustin de Coulomb (1736-1806). 219 */ 220 public static final Unit<ElectricCharge> COULOMB = addUnit(new AlternateUnit<ElectricCharge>(SECOND.multiply(AMPERE), "C"), ElectricCharge.class); 221 222 /** 223 * The SI unit for electric potential difference, electromotive force (standard name <code>V</code>). One Volt is equal to the difference of 224 * electric potential between two points on a conducting wire carrying a constant current of one ampere when the power dissipated between the points 225 * is one watt. It is named after the Italian physicist Count Alessandro Volta (1745-1827). 226 */ 227 public static final Unit<ElectricPotential> VOLT = addUnit(new AlternateUnit<ElectricPotential>(WATT.divide(AMPERE), "V"), ElectricPotential.class); 228 229 /** 230 * The SI unit for capacitance (standard name <code>F</code>). One Farad is equal to the capacitance of a capacitor having an equal and opposite 231 * charge of 1 coulomb on each plate and a potential difference of 1 volt between the plates. It is named after the British physicist and chemist 232 * Michael Faraday (1791-1867). 233 */ 234 public static final Unit<ElectricCapacitance> FARAD = addUnit(new AlternateUnit<ElectricCapacitance>(COULOMB.divide(VOLT), "F"), 235 ElectricCapacitance.class); 236 237 /** 238 * The SI unit for electric resistance (standard name <code>Ohm</code>). One Ohm is equal to the resistance of a conductor in which a current of one 239 * ampere is produced by a potential of one volt across its terminals. It is named after the German physicist Georg Simon Ohm (1789-1854). 240 */ 241 public static final Unit<ElectricResistance> OHM = addUnit(new AlternateUnit<ElectricResistance>(VOLT.divide(AMPERE), "Ω"), 242 ElectricResistance.class); 243 244 /** 245 * The SI unit for electric conductance (standard name <code>S</code>). One Siemens is equal to one ampere per volt. It is named after the German 246 * engineer Ernst Werner von Siemens (1816-1892). 247 */ 248 public static final Unit<ElectricConductance> SIEMENS = addUnit(new AlternateUnit<ElectricConductance>(AMPERE.divide(VOLT), "S"), 249 ElectricConductance.class); 250 251 /** 252 * The SI unit for magnetic flux (standard name <code>Wb</code>). One Weber is equal to the magnetic flux that in linking a circuit of one turn 253 * produces in it an electromotive force of one volt as it is uniformly reduced to zero within one second. It is named after the German physicist 254 * Wilhelm Eduard Weber (1804-1891). 255 */ 256 public static final Unit<MagneticFlux> WEBER = addUnit(new AlternateUnit<MagneticFlux>(VOLT.multiply(SECOND), "Wb"), MagneticFlux.class); 257 258 /** 259 * The alternate unit for magnetic flux density (standard name <code>T</code>). One Tesla is equal equal to one weber per square metre. It is named 260 * after the Serbian-born American electrical engineer and physicist Nikola Tesla (1856-1943). 261 */ 262 public static final Unit<MagneticFluxDensity> TESLA = addUnit(new AlternateUnit<MagneticFluxDensity>(WEBER.divide(METRE.pow(2)), "T"), 263 MagneticFluxDensity.class); 264 265 /** 266 * The alternate unit for inductance (standard name <code>H</code>). One Henry is equal to the inductance for which an induced electromotive force 267 * of one volt is produced when the current is varied at the rate of one ampere per second. It is named after the American physicist Joseph Henry 268 * (1791-1878). 269 */ 270 public static final Unit<ElectricInductance> HENRY = addUnit(new AlternateUnit<ElectricInductance>(WEBER.divide(AMPERE), "H"), 271 ElectricInductance.class); 272 273 /** 274 * The SI unit for Celsius temperature (standard name <code>Cel</code>). This is a unit of temperature such as the freezing point of water (at one 275 * atmosphere of pressure) is 0 Cel, while the boiling point is 100 Cel. 276 */ 277 @SuppressWarnings({ "rawtypes", "unchecked" }) 278 public static final Unit<Temperature> CELSIUS = addUnit(new TransformedUnit(KELVIN, new AddConverter(273.15))); 279 // Not mapping to Temperature since temperature is mapped to Kelvin. 280 281 /** 282 * The SI unit for luminous flux (standard name <code>lm</code>). One Lumen is equal to the amount of light given out through a solid angle by a 283 * source of one candela intensity radiating equally in all directions. 284 */ 285 public static final Unit<LuminousFlux> LUMEN = addUnit(new AlternateUnit<LuminousFlux>(CANDELA.multiply(STERADIAN), "lm"), LuminousFlux.class); 286 287 /** 288 * The SI unit for illuminance (standard name <code>lx</code>). One Lux is equal to one lumen per square metre. 289 */ 290 public static final Unit<Illuminance> LUX = addUnit(new AlternateUnit<Illuminance>(LUMEN.divide(METRE.pow(2)), "lx"), Illuminance.class); 291 292 /** 293 * The SI unit for activity of a radionuclide (standard name <code>Bq</code> ). One becquerel is the radiation caused by one disintegration per 294 * second. It is named after the French physicist, Antoine-Henri Becquerel (1852-1908). 295 */ 296 public static final Unit<Radioactivity> BECQUEREL = addUnit(new AlternateUnit<Radioactivity>(ONE.divide(SECOND), "Bq"), Radioactivity.class); 297 298 /** 299 * The SI unit for absorbed dose, specific energy (imparted), kerma (standard name <code>Gy</code>). One gray is equal to the dose of one joule of 300 * energy absorbed per one kilogram of matter. It is named after the British physician L. H. Gray (1905-1965). 301 */ 302 public static final Unit<RadiationDoseAbsorbed> GRAY = addUnit(new AlternateUnit<RadiationDoseAbsorbed>(JOULE.divide(KILOGRAM), "Gy"), 303 RadiationDoseAbsorbed.class); 304 305 /** 306 * The SI unit for dose equivalent (standard name <code>Sv</code>). One Sievert is equal is equal to the actual dose, in grays, multiplied by a 307 * "quality factor" which is larger for more dangerous forms of radiation. It is named after the Swedish physicist Rolf Sievert (1898-1966). 308 */ 309 public static final Unit<RadiationDoseEffective> SIEVERT = addUnit(new AlternateUnit<RadiationDoseEffective>(JOULE.divide(KILOGRAM), "Sv"), 310 RadiationDoseEffective.class); 311 312 /** 313 * The SI unit for catalytic activity (standard name <code>kat</code>). 314 */ 315 public static final Unit<CatalyticActivity> KATAL = addUnit(new AlternateUnit<CatalyticActivity>(MOLE.divide(SECOND), "kat"), 316 CatalyticActivity.class); 317 318 // //////////////////////////// 319 // SI DERIVED PRODUCT UNITS // 320 // //////////////////////////// 321 322 /** 323 * The SI unit for velocity quantities (standard name <code>m/s</code>). 324 */ 325 public static final Unit<Speed> METRE_PER_SECOND = addUnit(new ProductUnit<>(METRE.divide(SECOND)), Speed.class); 326 327 /** 328 * The SI unit for acceleration quantities (standard name <code>m/s2</code> ). 329 */ 330 public static final Unit<Acceleration> METRE_PER_SQUARE_SECOND = addUnit(new ProductUnit<>(METRE_PER_SECOND.divide(SECOND)), Acceleration.class); 331 332 /** 333 * The SI unit for area quantities (standard name <code>m2</code>). 334 */ 335 public static final Unit<Area> SQUARE_METRE = addUnit(new ProductUnit<>(METRE.multiply(METRE)), Area.class); 336 337 /** 338 * The SI unit for volume quantities (standard name <code>m3</code>). 339 */ 340 public static final Unit<Volume> CUBIC_METRE = addUnit(new ProductUnit<Volume>(SQUARE_METRE.multiply(METRE)), Volume.class); 341 342 /** 343 * A unit of velocity expressing the number of international {@link #KILOMETRE kilometres} per {@link #HOUR hour} (abbreviation <code>km/h</code>). 344 */ 345 public static final Unit<Speed> KILOMETRE_PER_HOUR = addUnit(METRE_PER_SECOND.multiply(0.277778d)).asType(Speed.class); 346 347 // /////////////////////////////////////////////////////////////// 348 // Common Units outside the SI that are accepted for use with the SI. // 349 // /////////////////////////////////////////////////////////////// 350 351 /** 352 * A dimensionless unit accepted for use with SI units (standard name <code>%</code>). 353 */ 354 public static final Unit<Dimensionless> PERCENT = addUnit(new TransformedUnit<>(ONE, new RationalConverter(1, 100))); 355 356 // //////// 357 // Time // 358 // //////// 359 /** 360 * A time unit accepted for use with SI units (standard name <code>min</code>). 361 */ 362 public static final Unit<Time> MINUTE = addUnit(new TransformedUnit<>("min", SECOND, SECOND, new RationalConverter(60, 1))); 363 364 /** 365 * A time unit accepted for use with SI units (standard name <code>h</code> ). 366 */ 367 public static final Unit<Time> HOUR = addUnit(new TransformedUnit<>("h", SECOND, SECOND, new RationalConverter(60 * 60, 1))); 368 369 /** 370 * A time unit accepted for use with SI units (standard name <code>d</code> ). 371 */ 372 public static final Unit<Time> DAY = addUnit(new TransformedUnit<>("d", SECOND, SECOND, new RationalConverter(24 * 60 * 60, 1))); 373 374 /** 375 * A unit of duration equal to 7 {@link #DAY} (standard name <code>week</code>). 376 */ 377 public static final Unit<Time> WEEK = addUnit(DAY.multiply(7)); 378 379 /** 380 * A time unit accepted for use with SI units (standard name <code>y</code> ). 381 */ 382 public static final Unit<Time> YEAR = addUnit(Units.DAY.multiply(365.2425)); 383 384 /** 385 * A volume unit accepted for use with SI units (standard name <code>l</code>). 386 * 387 * @see <a href="https://en.wikipedia.org/wiki/Litre"> Wikipedia: Litre</a> 388 */ 389 public static final Unit<Volume> LITRE = AbstractSystemOfUnits.Helper.addUnit(INSTANCE.units, new TransformedUnit<Volume>(CUBIC_METRE, 390 new RationalConverter(1, 1000)), "Litre", "l"); 391 392 /** 393 * Returns the unique instance of this class. 394 * 395 * @return the Units instance. 396 */ 397 public static SystemOfUnits getInstance() { 398 return INSTANCE; 399 } 400 401 static { 402 // have to add AbstractUnit.ONE as Dimensionless, too 403 addUnit(ONE); 404 INSTANCE.quantityToUnit.put(Dimensionless.class, ONE); 405 } 406 407 /** 408 * Adds a new unit not mapped to any specified quantity type. 409 * 410 * @param unit 411 * the unit being added. 412 * @return <code>unit</code>. 413 */ 414 private static <U extends Unit<?>> U addUnit(U unit) { 415 INSTANCE.units.add(unit); 416 return unit; 417 } 418 419 /** 420 * Adds a new unit and maps it to the specified quantity type. 421 * 422 * @param unit 423 * the unit being added. 424 * @param type 425 * the quantity type. 426 * @return <code>unit</code>. 427 */ 428 private static <U extends AbstractUnit<?>> U addUnit(U unit, Class<? extends Quantity<?>> type) { 429 INSTANCE.units.add(unit); 430 INSTANCE.quantityToUnit.put(type, unit); 431 return unit; 432 } 433}