mirror of
https://github.com/nasa/trick.git
synced 2024-12-27 00:31:07 +00:00
242 lines
12 KiB
HTML
242 lines
12 KiB
HTML
<html>
|
|
|
|
<head>
|
|
<meta name="GENERATOR" content="Microsoft FrontPage 5.0">
|
|
<meta name="ProgId" content="FrontPage.Editor.Document">
|
|
<meta http-equiv="Content-Type" content="text/html; charset=windows-1252">
|
|
<title>C++ Units</title>
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<p><b><font face="Arial"><font size="4">C++ Units</font><br>
|
|
</font></b><font face="Arial" size="2">by <a href="../index.html">Calum Grant</a></font></p>
|
|
<p><font face="Arial" size="2">Units home page: <a href="index.html">
|
|
http://calumgrant.net/units</a></font></p>
|
|
<p><b><font face="Arial">Introduction</font></b></p>
|
|
<blockquote>
|
|
<p><font face="Arial" size="2">This library is for engineers and scientists
|
|
who deal with physical quantities. Units provide a safety net that check
|
|
the validity of formulae at compile-time and ensure that different units are
|
|
converted where necessary. A large number of physical quantities are
|
|
provided by the library, including all SI units.</font></p>
|
|
<p><font face="Arial" size="2">This library is for software engineers.
|
|
By giving a quantity a unit, it prevents the wrong value being assigned to the
|
|
wrong thing, or passing the wrong value to the wrong argument of a function.
|
|
Units documentation a program - <i>int</i> is not descriptive, but <i>apples</i> is.</font></p>
|
|
<p><font face="Arial" size="2">The library uses templates and generative
|
|
programming techniques to handle units at compile time, so there is no
|
|
run-time overhead. Many units are predefined and it is straightforward
|
|
to add your own.</font></p>
|
|
<p><font face="Arial" size="2">The library is installed by copying the file
|
|
<a href="units.hpp">units.hpp</a> into your include directory. To use the library</font></p>
|
|
<blockquote>
|
|
<pre>#include "units.hpp"</pre>
|
|
</blockquote>
|
|
</blockquote>
|
|
<p><b><font face="Arial">Using built in units</font></b></p>
|
|
<blockquote>
|
|
<p><font face="Arial" size="2">The built in physical quantities are in the
|
|
namespace <i>units::values</i>, so for the purposes of this tutorial, assume
|
|
that</font></p>
|
|
<blockquote>
|
|
<pre>using namespace units::values;</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">has been written somewhere near the top of the
|
|
file. This namespace gives you access to the following physical
|
|
quantities:</font></p>
|
|
<ul>
|
|
<li><font face="Arial" size="2">No unit: <i>unit, percent, dozen,
|
|
bakers_dozen</i></font></li>
|
|
<li><font face="Arial" size="2">Mass: <i>kg, g, mg, lb, oz, tonne</i></font></li>
|
|
<li><font face="Arial" size="2">Time: <i>s, ms, minute, hour, day, week</i></font></li>
|
|
<li><font face="Arial" size="2">Calendar: <i>month, year, century,
|
|
millennium</i></font></li>
|
|
<li><font face="Arial" size="2">Distance: <i>m, cm, mm, km, inch, foot,
|
|
yard, mile, nautical_mile</i></font></li>
|
|
<li><font face="Arial" size="2">Temperature: <i>K, Celsius, Fahrenheit</i></font></li>
|
|
<li><font face="Arial" size="2">Force: <i>N</i></font></li>
|
|
<li><font face="Arial" size="2">Pressure: <i>Pa, kPa, psi, millibar</i></font></li>
|
|
<li><font face="Arial" size="2">Energy: <i>J</i></font></li>
|
|
<li><font face="Arial" size="2">Power: <i>W</i></font></li>
|
|
<li><font face="Arial" size="2">Area: <i>m2, hectare, are, inch2, acre</i></font></li>
|
|
<li><font face="Arial" size="2">Volume: <i>cm3, ml, cl, liter, dl, m3</i></font></li>
|
|
<li><font face="Arial" size="2">Velocity: <i>mph, kph, meters_per_second,
|
|
knot, mach</i></font></li>
|
|
<li><font face="Arial" size="2">Angle: <i>rad, degree, grad, degree_minute,
|
|
degree_second</i></font></li>
|
|
<li><font face="Arial" size="2">Other units: <i>A, mol, cd, rad, sr, Hz, C,
|
|
V, F, Ohm, S, Wb, T, H, lm, lx, Bq, Gy, Sv, kat, rpm</i></font></li>
|
|
</ul>
|
|
<p><font size="2" face="Arial">The naming convention is to use the symbol of
|
|
the SI unit. For other units, the name is used since there are no
|
|
standard symbols.</font></p>
|
|
<p><font face="Arial" size="2">The built-in values are all doubles, however
|
|
you can change this (see next section). The physical quantities behave
|
|
exactly as normal doubles, except that they check the units and perform
|
|
conversion where necessary.</font></p>
|
|
<p><font face="Arial" size="2">A value can be constructed from a number,
|
|
another value, or the default constructor which initializes the value to zero.
|
|
When constructing from another value, the other value must be compatible or
|
|
you will get a compile-time error:</font></p>
|
|
<blockquote>
|
|
<pre>m x; // x = 0
|
|
m y(10); // y = 10
|
|
m z(cm(250)); // z = 2.5
|
|
m w(s(3)); // Compile-time error: incompatible units
|
|
|
|
kph(mph(70)) // Convert miles per hour to km per hour</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">Assignment is only possible from another value,
|
|
and the value must be compatible:</font></p>
|
|
<blockquote>
|
|
<pre>x = 10; // Compile-time error: 10 of what?
|
|
x = cm(200); // Ok: x = 2.0
|
|
x = s(3) // Compile-time error: incompatible units</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The number in the value can be obtained using
|
|
the <i>get() </i>method:</font></p>
|
|
<blockquote>
|
|
<pre>x.get(); // Get current value of x
|
|
cm(m(5)).get() // 500</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The normal arithmetic operators are supported:
|
|
<i>+, -, *, /, /=, *=, ++, --</i>. There are some restrictions however.
|
|
You can only add or subtract compatible values. You can multiply or
|
|
divide by any other value, which returns a value with a unit with the correct
|
|
type (multiplied or divided). The <i>*=</i> and <i>/=</i> operators can
|
|
only take a number (in general if you multiply by another unit the type will
|
|
be different). e.g.</font></p>
|
|
<blockquote>
|
|
<pre>miles x = m(1) + cm(10); // Add two distances
|
|
m2 area = foot(10) * yard(3); // Converts to meters squared
|
|
s timer;
|
|
timer += 10; // Compile-time error: add 10 of what
|
|
timer += s(10); // Ok: add 10 seconds to timer
|
|
liter v = hectare(1) * mm(1); // Ok
|
|
H h( m(4)*m(8)*kg(2)/s(2)/s(4)/A(1)/A(2) ); // A complex (but valid) formula
|
|
s(10) + m(4); // Compile-time error: Can't add time and distance</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The comparison operators (<i>==, !=, <, <=, >,
|
|
>=</i>) also convert between units implicitly:</font></p>
|
|
<blockquote>
|
|
<pre>minutes(2) > s(70) // true</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">Writing a value to a stream (using <i>operator
|
|
<<</i>) displays the unit after the value. Many of the built in units
|
|
have names, otherwise the text is generated, as in the following example:</font></p>
|
|
<pre> std::cout << "Flow rate is " << m3(mile(1)*inch(80)*foot(9))/s(minute(5));
|
|
// Output: Flow rate is 29.9026 (m)^3.(s)^-1</pre>
|
|
<p><font face="Arial" size="2">The <i>units::sqrt</i> function provides a
|
|
square root - which also takes the root of the unit of course.</font></p>
|
|
<blockquote>
|
|
<pre>m a, b, c;
|
|
c = units::sqrt( a*a + b*b );</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">You can take an arbitrary rational
|
|
power of a number using the <i>units::raise</i> template, however you must
|
|
specify the power at compile time so that the compiler knows the unit of the return
|
|
value. e.g.</font></p>
|
|
<blockquote>
|
|
<pre>units::raise<3,1>(m(2)) == m(2)*m(2)*m(2)</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">There are some trigonometric functions (<i>units::sin</i>,
|
|
<i>units::cos</i> and <i>units::tan</i>) which take an angle. You can
|
|
supply any unit of angle to the function and the function will convert to
|
|
radians. e.g.</font></p>
|
|
<blockquote>
|
|
<pre>units::tan( degree(45) );</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">There are a number of constants available in
|
|
the <i>units::constants</i> namespace. These are</font></p>
|
|
<ul>
|
|
<li><i><font face="Arial" size="2">k, mu, NA, G0, e0, me, eV, e, F, alpha,
|
|
inv_alpha, u0, phi0, R, G, h, h_bar, mp, Rinf, c, rho, pi, lightyear, g</font></i></li>
|
|
</ul>
|
|
<p><font face="Arial" size="2">Example:</font></p>
|
|
<blockquote>
|
|
<pre>N attractive_force = units::constants::G * kg(1) * kg(2) / m(3) / m(5);</pre>
|
|
</blockquote>
|
|
</blockquote>
|
|
<p><b><font face="Arial">Creating new units</font></b></p>
|
|
<blockquote>
|
|
<pre><font face="sans-serif" size="2">Values with units are provided by the <i>units::value<></i> class template, declared as follows:</font> </pre>
|
|
<blockquote>
|
|
<pre>template<typename Value, typename Unit>
|
|
class value; </pre>
|
|
</blockquote>
|
|
<p><font face="sans-serif" size="2">The predefined units (as used in the
|
|
previous section) are declared in the <i>units::units</i> namespace. So
|
|
you can reuse any of these units but provide a different type, e.g.</font></p>
|
|
<blockquote>
|
|
<pre>units::value<float, units::units::m> length;</pre>
|
|
</blockquote>
|
|
<p><font face="sans-serif" size="2">The unit can be any type, so to create a
|
|
new unit, just create a new type:</font> </p>
|
|
<blockquote>
|
|
<pre>struct apples;
|
|
struct oranges;
|
|
typedef units::value<int, apples> apples_t;
|
|
typedef units::value<int, oranges> oranges_t; </pre>
|
|
</blockquote>
|
|
<pre><font face="sans-serif" size="2">The new value will be protected from interoperating with naked numbers or other types of unit:</font></pre>
|
|
<blockquote>
|
|
<pre>apples_t n(5);
|
|
n = oranges_t(3); // Compile-time error
|
|
n = apples_t(10); // Ok </pre>
|
|
</blockquote>
|
|
</blockquote>
|
|
<p><b><font face="sans-serif">Displaying units</font> </b></p>
|
|
<blockquote>
|
|
<p><font face="sans-serif" size="2">When a value is output (using <i>
|
|
operator<<</i>), it appends the unit to the stream. By default the unit will
|
|
display the text <i>"units"</i>. You can change this by declaring the name of the
|
|
unit with the <i>UNITS_DISPLAY_NAME</i> macro, which takes the unit as its
|
|
first parameter, and its name as its second parameter. e.g.</font> </p>
|
|
<blockquote>
|
|
<pre>UNITS_DISPLAY_NAME( apples, "apples" );
|
|
UNITS_DISPLAY_NAME( oranges, "oranges" ); </pre>
|
|
<pre>std::cout << oranges_t(2); // Output: 2 oranges</pre>
|
|
</blockquote>
|
|
</blockquote>
|
|
<p><b><font face="sans-serif">Converting between units</font> </b></p>
|
|
<blockquote>
|
|
<p><font face="sans-serif" size="2">A unit can be defined in terms of other
|
|
units. The templates <i>units::scale, units::translate, units::pow</i> and <i>
|
|
units::compose</i> provide a means of constructing new units which can be
|
|
converted from and to another unit.</font></p>
|
|
<p><font face="sans-serif" size="2">The <i>units::scale<></i> template
|
|
constructs a unit which is a multiple of another:</font></p>
|
|
<blockquote>
|
|
<pre>struct penny;
|
|
typedef units::scale<penny, 4> farthing; // Multiply penny * 4 to get farthings
|
|
typedef units::scale<penny, 1, 12> shilling; // Multiply penny by 1/12 to get shillings
|
|
typedef units::scale<shilling, 1, 20> pound;
|
|
typedef units::scale<penny, 1, 30> half_crown;
|
|
typedef units::scale<half_crown, 1, 2> crown;</pre>
|
|
<pre>std::cout << "There are " << penny(pound(1)).get() << " old pence in the pound\n";</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The <i>units::translate<></i> template
|
|
constructs a unit which is offset from another:</font></p>
|
|
<blockquote>
|
|
<pre>typedef units::translate<units::units::K, -27315, 100> Celcius;
|
|
// Celcius = K - 27315/100</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The <i>units::pow<></i> template constructs a unit
|
|
which is a power of another unit:</font></p>
|
|
<blockquote>
|
|
<pre>typedef units::pow<units::units::m, 3> m3; // One cubic meter</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">Finally the <i>units::compose<></i> template
|
|
creates a unit which multiplies two other units:</font></p>
|
|
<blockquote>
|
|
<pre>typedef units::compose<power, time> energy;
|
|
typedef units::compose< units::units::m, units::pow<units::units::s, -1> > meters_per_second;</pre>
|
|
</blockquote>
|
|
<p><font face="Arial" size="2">The conversion operators are able to analyse
|
|
the type of the unit and generate a conversion function automatically!</font></p>
|
|
</blockquote>
|
|
|
|
</body>
|
|
|
|
</html> |