This tutorial will guide you in the basic use of ThermoState. The ThermoState package is designed to ease the evaluation of thermodynamic properties for common substances used in Mechanical Engineering courses. Rather than looking up the information in a table and interpolating, we can input properties for the states directly, and all unknown values are automatically calculated.
In [ ]:
from thermostate import State, Q_, units
Now that the interface has been imported, we can create some properties. For instance, let's say we're given the pressure and temperature properties for water, and asked to determine the specific volume. First, let's create variables that set the pressure and temperature. We will use the Pint Quantity
function, which we have called Q_
. The syntax for the Q_
function is Q_(value, 'units')
.
In [ ]:
p_1 = Q_(101325, 'Pa')
We can use whatever units we'd like, Pint supports a wide variety of units.
In [ ]:
p_1 = Q_(1.01325, 'bar')
p_1 = Q_(14.7, 'psi')
p_1 = Q_(1.0, 'atm')
Another way to specify the units is to use the units
class that we imported. This class has a number of attributes (text following a period) that can be used to create a quantity with units by multiplying a number with the unit.
units.degR
# ^^^^
# This is the attribute
Let's set the temperature now. The available units of temperature are degF
(fahrenheit
), degR
(rankine
), degC
(celsius
), and K
(kelvin
).
In [ ]:
T_1 = 460*units.degR
T_1 = 25*units.degC
T_1 = 75*units.degF
T_1 = 400*units.K
The two ways of creating the units are equivalent. The following cell should print True
to demonstrate this.
In [ ]:
Q_(101325, 'Pa') == 1.0*units.atm
Note the convention we are using here: the variables are named with the property, followed by an underscore, then the number of the state. In this case, we are setting properties for state 1, hence T_1
and p_1
.
Now that we have defined two properties with units, let's define a state. First, we create a variable to hold the State
and tell ThermoState what substance we want to use with that state. The available substances are:
water
air
R134a
R22
propane
ammonia
isobutane
carbondioxide
oxygen
nitrogen
Note that the name of the substance is case-insensitive (it doesn't matter whether you use lower case or upper case). It is often easiest to set the name of the substance in a variable, like:
In [ ]:
substance = 'water'
Now we need to create the State
and assign values for the properties. Properties of the state are set as arguments to the State
class, and they must always be set in pairs, we cannot set a single property at a time. The syntax is
st = State(substance, property_1=value, property_2=value)
To demonstrate, we will set the T
and p
properties of the state and set them equal to the temperature and pressure we defined above. Note that the capitalization of the properties is important! The p
is lower case while the T
is upper case (lower case t
means time).
In [ ]:
print('T = {}, p = {}'.format(T_1, p_1))
In [ ]:
st_1 = State(substance, T=T_1, p=p_1)
Note again the convention we are using here: The state is labeled by st
, then an underscore, then the number of the state.
The variables that we use on the right side of the equal sign in the State
function can be named anything we want. For instance, the following code is exactly equivalent to what we did before.
In [ ]:
luke = Q_(1.0, 'atm')
leia = Q_(400.0, 'K')
print('Does luke equal p_1?', luke == p_1)
print('Does leia equal T_1?', leia == T_1)
st_starwars = State(substance, T=leia, p=luke)
print('Does st_starwars equal st_1?', st_starwars == st_1)
In theory, any two pairs of independent properties can be used to set the state. In reality, the pairs of properties available to set the state is slightly limited because of the way the equation of state is written. The available pairs of properties are
Tp
Ts
Tv
Tx
pu
ps
pv
ph
px
uv
sv
hs
hv
The reverse of any of these pairs is also possible and totally equivalent.
By setting two properties in this way, the State
class will calculate all the other properties we might be interested in. We can access the value of any property by getting the attribute for that property. The available properties are T
(temperature), p
(pressure), v
(mass-specific volume), u
(mass-specific internal energy), h
(mass-specific enthalpy), s
(mass-specific entropy), x
(quality), cp
(specific heat at constant pressure), cv
(specific heat at constant volume), and phase
(the phase of this state). The syntax is
State.property
or
st_1.T # Gets the temperature
st_1.p # Gets the pressure
st_1.v # Gets the specific volume
st_1.u # Gets the internal energy
st_1.h # Gets the enthalpy
st_1.s # Gets the entropy
st_1.x # Gets the quality
st_1.cp # Gets the specific heat at constant pressure
st_1.cv # Gets the specific heat at constant volume
st_1.phase # Gets the phase at this state
In [ ]:
print('T_1 = {}'.format(st_1.T))
print('p_1 = {}'.format(st_1.p))
print('v_1 = {}'.format(st_1.v))
print('u_1 = {}'.format(st_1.u))
print('h_1 = {}'.format(st_1.h))
print('s_1 = {}'.format(st_1.s))
print('x_1 = {}'.format(st_1.x))
print('cp_1 = {}'.format(st_1.cp))
print('cv_1 = {}'.format(st_1.cv))
print('phase_1 = {}'.format(st_1.phase))
In this case, the value for the quality is the special Python value None
. This is because at 400 K and 101325 Pa, the state of water is a superheated vapor and the quality is undefined except in the vapor dome. To access states in the vapor dome, we cannot use T
and p
as independent properties, because they are not independent inside the vapor dome. Instead, we have to use the pairs involving the other properties (possibly including the quality) to set the state. When we define the quality, the units are dimensionless
or percent
. For instance:
In [ ]:
T_2 = Q_(100.0, 'degC')
x_2 = Q_(0.1, 'dimensionless')
st_2 = State('water', T=T_2, x=x_2)
print('T_2 = {}'.format(st_2.T))
print('p_2 = {}'.format(st_2.p))
print('v_2 = {}'.format(st_2.v))
print('u_2 = {}'.format(st_2.u))
print('h_2 = {}'.format(st_2.h))
print('s_2 = {}'.format(st_2.s))
print('x_2 = {}'.format(st_2.x))
In addition, whether you use the 'dimensionless'
"units" for the quality as above, or use the 'percent'
"units", the result is exactly equivalent. The next cell should print True
to the screen to demonstrate this.
In [ ]:
x_2 == Q_(10.0, 'percent')
From these results, we can see that the units of the units of the properties stored in the State
are always SI units - Kelvin, Pascal, m3/kg, J/kg, and J/(kg-Kelvin). We can use the to
function to convert the units to anything we want, provided the dimensions are compatible. The syntax is State.property.to('units')
.
In [ ]:
print(st_2.T.to('degF'))
print(st_2.s.to('BTU/(lb*degR)'))
If we try to convert to a unit with incompatible dimensions, Pint will raise a DimenstionalityError
exception.
In [ ]:
print(st_2.T.to('joule'))
Here we have tried to convert from 'kelvin'
to 'joule'
and the error message which is the last line says
DimensionalityError: Cannot convert from 'kelvin' ([temperature]) to 'joule' ([length] ** 2 * [mass] / [time] ** 2)
The dimensions of a temperature are, well, temperature. The formula for energy (Joule) is $m*a*d$ (mass times acceleration times distance), and in terms of dimensions, $M*L/T^2*L = L^2*M/T^2$ (where in dimensions, capital $T$ is time). Clearly, these dimensions are incompatible. A more subtle case might be trying to convert energy to power (again, not allowed):
In [ ]:
Q_(1000.0, 'joule').to('watt')
Other common errors generated from using ThermoState will raise StateErrors
. These errors may be due to
Tu
, Th
, and us
, due to limitations in CoolProp)Quantity
with incorrect dimensions for the property inputAn example demonstrating #4 from above:
In [ ]:
State('water', v=Q_(1000.0, 'degC'), p=Q_(1.0, 'bar'))
In summary, we need to use two (2) independent and intensive properties to fix the state of any simple compressible system. We need to define these quantities with units using Pint, and then use them to set the conditions of a State. Then, we can access the other properties of the State by using the attributes.
In [ ]:
h_3 = Q_(2000.0, 'kJ/kg')
s_3 = Q_(3.10, 'kJ/(kg*K)')
st_3 = State('water', h=h_3, s=s_3)
print('T_3 = {}'.format(st_3.T))
print('p_3 = {}'.format(st_3.p))
print('v_3 = {}'.format(st_3.v))
print('u_3 = {}'.format(st_3.u))
print('h_3 = {}'.format(st_3.h))
print('s_3 = {}'.format(st_3.s))
print('x_3 = {}'.format(st_3.x))