-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathpsychropy.py
441 lines (358 loc) · 15.8 KB
/
psychropy.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
''' This is a translated version of the visual basic psych function in
our xls template to a python code function psychropy()
The function and subfunctions defined herein come with no warranty or
certification of fitness for any purpose
Do not use these functions for conditions outside boundaries defined
by their original sources.
Subfunctions use equations from the following sources:
ASHRAE Fundamentals, 2005, SI Edition
Singh et al. "Numerical Calculations of Psychrometric Properties
on a Calculator". Building and Environment, 37, 2002.
The function will calculate various properties of moist air. Properties
calculated include Wet Bulb, Dew Point, Relative Humidity, Humidity
Ratio, Vapor Pressure, Degree of Saturation, enthalpy, specific volume
of dry air, and moist air density.
The function requires input of: barometric pressure, and two other
parameters, We recomend that one of these be Tdb and if not using that
the other two must be h and HR. These parameters along with Tdb can
be Twb, DP, RH, or two mentioned previously.
Sytax for function as follows:
psych(P,intype0,invalue0,intype1,invalue1,outtype,unittype)
Where:
P is the barometric pressure in PSI or Pa .
intypes indicator string for the corresponding
invalue parameter (ie Tdb, RH etc.)
invalues is the actual value associated with the type of parameter
(ie value of Wet bulb, Dew point, RH, or Humidity
Ratio etc.)
outType indicator string for the corresponding invalue parameter
unittype is the optional unit selector. Imp for Imperial, SI for
SI. Imp is default if omitted.
valid intypes:
Tdb Dry Bulb Temp F or C Valid for Input
*** it is highly Recommended Tdb be used as an input (can only
output/not use, if both other inputs are h and HR)
Twb Web Bulb Temp F or C Valid for Input
DP Dew point F or C Valid for input
RH RH between 0 and 1 Valid for input
W Humidity Ratio Mass Water/ Mass Dry Air Valid for input
h Enthalpy BTU/lb dry air or kJ/kg DA Valid for input
***Warning 0 state for Imp is ~0F, 0% RH ,and 1 ATM, 0 state
for SI is 0C, 0%RH and 1 ATM
valid outtypes:
Tdb Dry Bulb Temp F or C Valid for Input
***it is highly Recommended Tdb be used as an input (can only
output/not use, if both other inputs are h and HR)
Twb Web Bulb Temp F or C Valid for Input
DP Dew point F or C Valid for input
RH Relative Humidity between 0 and 1 Valid for input
W Humidity Ratio Mass Water/ Mass Dry Air Valid for input
h Enthalpy BTU/lb dry air or kJ/kg DA Valid for input
***Warning 0 state for Imp is ~0F, 0% RH ,and 1 ATM, 0 state
for SI is 0C, 0%RH and 1 ATM
WVP Water Vapor Pressure PSI or Pa
Dsat Degree of Saturation between 0 and 1
s NOT VALID, Should be entropy
SV Specific Volume ft^3/lbm or m^3/kg dry air
MAD Moist Air Density lb/ft^3 or m^3/kg
The corresponding numbers associated with the types in the excel VB program:
The Numbers for inType and outType are
1 Web Bulb Temp F or C Valid for Input
2 Dew point F or C Valid for input
3 RH between 0 and 1 Valid for input
4 Humidity Ratio Mass Water/ Mass Dry Air Valid for input
5 Water Vapor Pressure PSI or Pa
6 Degree of Saturation between 0 and 1
7 Enthalpy BTU/lb dry air or kJ/kg dry air
Warning 0 state for IP is ~0F, 0% RH ,and 1 ATM, 0 state
for SI is 0C, 0%RH and 1 ATM
8 NOT VALID, Should be entropy
9 Specific Volume ft**3/lbm or m**3/kg dry air
10 Moist Air Density lb/ft**3 or m**3/kg"""
this python version adds the capability to use Enthalpy in the place of Tdb
modified Syntax to be psychro(p,in0type,in0,in1type,in1,outtype,units)
where p is pressure
in0type is the type of the first input variable
in0 is the value
in1type is the type of the first input variable
in1 is the value
outtype is the type for the output variable
units is the specified units (ie Imp or SI)
'''
import math
def Part_press(P,W):
''' Function to compute partial vapor pressure in [kPa]
From page 6.9 equation 38 in ASHRAE Fundamentals handbook (2005)
P = ambient pressure [kPa]
W = humidity ratio [kg/kg dry air]
'''
result = P * W / (0.62198 + W)
return result
def Sat_press(Tdb):
''' Function to compute saturation vapor pressure in [kPa]
ASHRAE Fundamentals handbood (2005) p 6.2, equation 5 and 6
Tdb = Dry bulb temperature [degC]
Valid from -100C to 200 C
'''
C1 = -5674.5359
C2 = 6.3925247
C3 = -0.009677843
C4 = 0.00000062215701
C5 = 2.0747825E-09
C6 = -9.484024E-13
C7 = 4.1635019
C8 = -5800.2206
C9 = 1.3914993
C10 = -0.048640239
C11 = 0.000041764768
C12 = -0.000000014452093
C13 = 6.5459673
TK = Tdb + 273.15 # Converts from degC to degK
if TK <= 273.15:
result = math.exp(C1/TK + C2 + C3*TK + C4*TK**2 + C5*TK**3 +
C6*TK**4 + C7*math.log(TK)) / 1000
else:
result = math.exp(C8/TK + C9 + C10*TK + C11*TK**2 + C12*TK**3 +
C13*math.log(TK)) / 1000
return result
def Hum_rat(Tdb, Twb, P):
''' Function to calculate humidity ratio [kg H2O/kg air]
Given dry bulb and wet bulb temp inputs [degC]
ASHRAE Fundamentals handbood (2005)
Tdb = Dry bulb temperature [degC]
Twb = Wet bulb temperature [degC]
P = Ambient Pressure [kPa]
'''
Pws = Sat_press(Twb)
Ws = 0.62198 * Pws / (P - Pws) # Equation 23, p6.8
if Tdb >= 0: # Equation 35, p6.9
result = (((2501 - 2.326*Twb)*Ws - 1.006*(Tdb - Twb))/
(2501 + 1.86*Tdb - 4.186*Twb))
else: # Equation 37, p6.9
result = (((2830 - 0.24*Twb)*Ws - 1.006*(Tdb - Twb))/
(2830 + 1.86*Tdb - 2.1*Twb))
return result
def Hum_rat2(Tdb, RH, P):
''' Function to calculate humidity ratio [kg H2O/kg air]
Given dry bulb and wet bulb temperature inputs [degC]
ASHRAE Fundamentals handbood (2005)
Tdb = Dry bulb temperature [degC]
RH = Relative Humidity [Fraction or %]
P = Ambient Pressure [kPa]
'''
Pws = Sat_press(Tdb)
result = 0.62198*RH*Pws/(P - RH*Pws) # Equation 22, 24, p6.8
return result
def Rel_hum(Tdb, Twb, P):
''' Calculates relative humidity ratio
ASHRAE Fundamentals handbood (2005)
Tdb = Dry bulb temperature [degC]
Twb = Wet bulb temperature [degC]
P = Ambient Pressure [kPa]
'''
W = Hum_rat(Tdb, Twb, P)
result = Part_press(P, W) / Sat_press(Tdb) # Equation 24, p6.8
return result
def Rel_hum2(Tdb, W, P):
''' Calculates the relative humidity given:
Tdb = Dry bulb temperature [degC]
P = ambient pressure [kPa]
W = humidity ratio [kg/kg dry air]
'''
Pw = Part_press(P, W)
Pws = Sat_press(Tdb)
result = Pw / Pws
return result
def Wet_bulb(Tdb, RH, P):
''' Calculates the Wet Bulb temp given:
Tdb = Dry bulb temperature [degC]
RH = Relative humidity ratio [Fraction or %]
P = Ambient Pressure [kPa]
Uses Newton-Rhapson iteration to converge quickly
'''
W_normal = Hum_rat2(Tdb, RH, P)
result = Tdb
' Solves to within 0.001% accuracy using Newton-Rhapson'
W_new = Hum_rat(Tdb, result, P)
while abs((W_new - W_normal) / W_normal) > 0.00001:
W_new2 = Hum_rat(Tdb, result - 0.001, P)
dw_dtwb = (W_new - W_new2) / 0.001
result = result - (W_new - W_normal) / dw_dtwb
W_new = Hum_rat(Tdb, result, P)
return result
def Enthalpy_Air_H2O(Tdb, W):
''' Calculates enthalpy in kJ/kg (dry air) given:
Tdb = Dry bulb temperature [degC]
W = Humidity Ratio [kg/kg dry air]
Calculations from 2005 ASHRAE Handbook - Fundamentals - SI P6.9 eqn 32
'''
result = 1.006*Tdb + W*(2501 + 1.86*Tdb)
return result
def T_drybulb_calc(h,W):
''' Calculates dry bulb Temp in deg C given:
h = enthalpy [kJ/kg K]
W = Humidity Ratio [kg/kg dry air]
back calculated from enthalpy equation above
***Warning 0 state for Imp is ~0F, 0% RH ,and 1 ATM, 0 state
for SI is 0C, 0%RH and 1 ATM
'''
result = (h-(2501*W))/(1.006+(1.86*W))
return result
def Dew_point(P, W):
''' Function to compute the dew point temperature (deg C)
From page 6.9 equation 39 and 40 in ASHRAE Fundamentals handbook (2005)
P = ambient pressure [kPa]
W = humidity ratio [kg/kg dry air]
Valid for Dew Points less than 93 C
'''
C14 = 6.54
C15 = 14.526
C16 = 0.7389
C17 = 0.09486
C18 = 0.4569
Pw = Part_press(P, W)
alpha = math.log(Pw)
Tdp1 = C14 + C15*alpha + C16*alpha**2 + C17*alpha**3 + C18*Pw**0.1984
Tdp2 = 6.09 + 12.608*alpha + 0.4959*alpha**2
if Tdp1 >= 0:
result = Tdp1
else:
result = Tdp2
return result
def Dry_Air_Density(P, Tdb, W):
''' Function to compute the dry air density (kg_dry_air/m**3), using pressure
[kPa], temperature [C] and humidity ratio
From page 6.8 equation 28 ASHRAE Fundamentals handbook (2005)
[rho_dry_air] = Dry_Air_Density(P, Tdb, w)
Note that total density of air-h2o mixture is:
rho_air_h2o = rho_dry_air * (1 + W)
gas constant for dry air
'''
R_da = 287.055
result = 1000*P/(R_da*(273.15 + Tdb)*(1 + 1.6078*W))
return result
def psych(P,in0Type,in0Val,in1Type,in1Val, outType,unitType='Imp'):
if in0Type != 'h' and in0Type != 'W' and in0Type != 'Tdb':
outVal = 'NAN'
elif in0Type == in1Type:
outVal = 'NAN'
if unitType == 'SI':
P=P/1000 # converts P to kPa
if in0Type == 'Tdb': # assign the first input
Tdb=in0Val
elif in0Type =='W':
W=in0Val
elif in0Type =='h':
h=in0Val
if in1Type == 'Tdb': # assign the second input
Tdb=in1Val
elif in1Type == 'Twb':
Twb=in1Val
elif in1Type =='DP':
Dew=in1Val
elif in1Type =='RH':
RH=in1Val
elif in1Type =='W':
W=in1Val
elif in1Type =='h':
h=in1Val
else: # converts to SI if not already
P = (P*4.4482216152605)/(0.0254**2*1000)
if in0Type =='Tdb':
Tdb = (in0Val-32)/1.8
elif in0Type == 'W':
W = in0Val
elif in0Type == 'h':
h = ((in0Val * 1.055056)/0.45359237) - 17.884444444
if in1Type == 'Tdb':
Tdb = (in1Val - 32)/1.8
elif in1Type == 'Twb':
Twb = (in1Val - 32)/1.8
elif in1Type == 'DP':
Dew = (in1Val - 32)/1.8
elif in1Type == 'RH':
RH = in1Val
elif in1Type == 'W':
W = in1Val
elif in1Type == 'h':
h = ((in1Val*1.055056)/0.45359237) - 17.884444444
if (in0Type == 'h' and in1Type == 'W'): # calculate Tdb if not given
Tdb = T_drybulb_calc(h,W)
elif (in0Type == 'W' and in1Type == 'h'):
Tdb = T_drybulb_calc(h,W)
if outType == 'RH' or outType == 'Twb': # Find RH
if in1Type == 'Twb': # given Twb
RH = Rel_hum(Tdb, Twb, P)
elif in1Type == 'DP': # given Dew
RH = Sat_press(Dew)/Sat_press(Tdb)
# elif in1Type == 'RH': # given RH
# RH already Set
elif in1Type == 'W': # given W
RH = Part_press(P, W) / Sat_press(Tdb)
elif in1Type == 'h':
W = (1.006 * Tdb - h) / (-(2501 + 1.86 * Tdb))
RH = Part_press(P, W) / Sat_press(Tdb)
else:
if in0Type != 'W': # Find W
if in1Type == 'Twb': # Given Twb
W = Hum_rat(Tdb, Twb, P)
elif in1Type == 'DP': # Given Dew
W = 0.621945*Sat_press(Dew)/(P - Sat_press(Dew))
' Equation taken from eq 20 of 2009 Fundemental chapter 1'
elif in1Type == 'RH': # Given RH
W = Hum_rat2(Tdb, RH, P)
# elif in1Type == 'W': # Given W
# W already known
elif in1Type == 'h': # Given h
W = (1.006*Tdb - h)/(-(2501 + 1.86*Tdb))
' Algebra from 2005 ASHRAE Handbook - Fundamentals - SI P6.9 eqn 32'
else:
print ('invalid input varilables')
# P, Tdb, and W are now availible
if outType == 'Tdb':
outVal = Tdb
elif outType == 'Twb': # Request Twb
outVal = Wet_bulb(Tdb, RH, P)
elif outType == 'DP': # Request Dew
outVal = Dew_point(P, W)
elif outType == 'RH': # Request RH
outVal = RH
elif outType == 'W': # Request W
outVal = W
elif outType == 'WVP': # Request Pw
outVal = Part_press(P, W) * 1000
elif outType == "DSat": # Request deg of sat
outVal = W / Hum_rat2(Tdb, 1, P)
'the middle arg of Hum_rat2 is suppose to be RH. RH is suppose to be 100%'
elif outType == 'h': # Request enthalpy
outVal = Enthalpy_Air_H2O(Tdb, W)
elif outType == 's': # Request entropy
outVal = 5 / 0
'don\'t have equation for Entropy, so I divided by zero to induce an error.'
elif outType == 'SV': # Request specific volume
outVal = 1 / (Dry_Air_Density(P, Tdb, W))
elif outType == 'MAD': # Request density
outVal = Dry_Air_Density(P, Tdb, W) * (1 + W)
if unitType == 'Imp': # Convert to Imperial units
if outType == 'Tdb' or outType == 'Twb' or outType == 'DP':
outVal = 1.8*outVal + 32
elif outType == 'WVP':
outVal = outVal*0.0254**2/4.448230531
elif outType == 'h':
outVal = (outVal + 17.88444444444)*0.45359237/1.055056
elif outType == 'SV':
outVal = outVal*0.45359265/((12*0.0254)**3)
elif outType == 'MAD':
outVal = outVal*(12*0.0254)**3/0.45359265
return outVal
def main():
P = input ("enter the value for P:")
in0Type = input ( "enter the first input type:")
in0Val = input("enter first input value:")
in1Type = input ( "enter the second input type:")
in1Val = input ( "enter second input value:")
outType = input ( "enter the output type:")
unitType = input ( "enter the unit type (Imp or SI):")
result=psych(float(P),in0Type,float(in0Val),in1Type,float(in1Val),outType,unitType)
print (result)
if __name__=='__main__': main()