-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathWetBulb.py
401 lines (336 loc) · 15.9 KB
/
WetBulb.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
# [Twb,Teq,epott]=WetBulb(TemperatureC,Pressure,Humidity,[HumidityMode])
#
# Calculate wet-bulb temperature, equivalent temperature, and equivalent
# potential temperature from temperature, pressure, and relative or
# specific humidity.
#
# METHODOLOGY:
#
# Calculates Wet Bulb Temperature, Theta_wb, Theta_e, Moist Pot Temp,
# Lifting Cond Temp, and Equiv Temp using Davies-Jones 2008 Method.
# 1st calculates the lifting cond temperature (Bolton 1980 eqn 22).
# Then calculates the moist pot temp (Bolton 1980 eqn 24). Then
# calculates Equivalent Potential Temperature (Bolton 1980 eqn 39).
# From equivalent pot temp, equiv temp and Theta_w (Davies-Jones
# 2008 eqn 3.5-3.8). An accurate 'first guess' of wet bulb temperature
# is determined (Davies-Jones 2008 eqn 4.8-4.11). Newton-Raphson
# is used for 2 iterations, determining final wet bulb temperature
# (Davies-Jones 2008 eqn 2.6).
#
# Reference: Bolton: The computation of equivalent potential temperature.
# Monthly weather review (1980) vol. 108 (7) pp. 1046-1053
# Davies-Jones: An efficient and accurate method for computing the
# wet-bulb temperature along pseudoadiabats. Monthly Weather Review
# (2008) vol. 136 (7) pp. 2764-2785
# Flatau et al: Polynomial fits to saturation vapor pressure.
# Journal of Applied Meteorology (1992) vol. 31 pp. 1507-1513
# Note: Pressure needs to be in mb, mixing ratio needs to be in
# kg/kg in some equations, and in g/kg in others.
# Calculates Iteration via Newton-Raphson Method. Only 2 iterations.
# Reference: Davies-Jones: An efficient and accurate method for computing the
# wet-bulb temperature along pseudoadiabats. Monthly Weather Review
# (2008) vol. 136 (7) pp. 2764-2785
# Flatau et al: Polynomial fits to saturation vapor pressure.
# Journal of Applied Meteorology (1992) vol. 31 pp. 1507-1513
# Note: Pressure needs to be in mb, mixing ratio needs to be in
# kg/kg in some equations.
#
# Ported from HumanIndexMod 04-08-16 by Jonathan R Buzan
# MATLAB port by Robert Kopp
#
# Last updated by Robert Kopp, robert-dot-kopp-at-rutgers-dot-edu, Wed Jun 08 18:03:02 EDT 2016
#
# Ported to Python by Xianxiang Li
# February 2, 2019
#
# March 24, 2022
# Small bug fix from Manuel Theurl: Fixed that iteration has been done until maxiter all the time
# -> much faster code execution
import numpy as np
SHR_CONST_TKFRZ = 273.15
def calc_RH_from_T_Td(T, Td, mode=0):
"""
Calculate relative humidity from dry-bulb temperature T (in degC) and dew-point temperature
Td (in degC), based on equation (4) and (5) (mode=1) or equations (2) and (3) (mode = 1) from
http://www.npl.co.uk/reference/faqs/how-do-i-convert-between-units-of-dew-point-and-relative-humidity-(faq-thermal)
Inputs:
T: dry-bulb temperature (in degC)
Td: dew-point temperature (in degC)
mode: the mode to calculate RH
mode = 0 (default): equations (4) and (5), which is more accurate but more complicated
mode = 1: equations (2) and (3), which is simpler
Outputs:
RH: relative humidity in %, range 0-100.
"""
if mode == 0:
Tk = T + SHR_CONST_TKFRZ
Tdk = Td + SHR_CONST_TKFRZ
es = np.exp( -6096.9385 * Tk**(-1) + 21.2409642 - 2.711193e-2 * Tk + \
1.673952e-5 * Tk**2.0 + 2.433502 * np.log(Tk))
e = np.exp( -6096.9385 * Tdk**(-1) + 21.2409642 - 2.711193e-2 * Tdk + \
1.673952e-5 * Tdk**2.0 + 2.433502 * np.log(Tdk))
elif mode == 1: # Magnus formulae
es = np.exp(np.log(611.2) + (17.62*T)/(243.12+T)) # vapor pressure in Pa
e = np.exp(np.log(611.2) + (17.62*Td)/(243.12+Td)) # vapor pressure in Pa
RH = e/es * 100.0
RH[RH>100] = 100.0
RH[RH<0] = 0
return RH
def QSat_2(T_k, p_t):
"""
[es_mb,rs,de_mbdT,dlnes_mbdT,rsdT,foftk,fdt]=QSat_2(T_k, p_t)
DESCRIPTION:
Computes saturation mixing ratio and the change in saturation
mixing ratio with respect to temperature. Uses Bolton eqn 10, 39.
Davies-Jones eqns 2.3,A.1-A.10
Reference: Bolton: The computation of equivalent potential temperature.
Monthly Weather Review (1980) vol. 108 (7) pp. 1046-1053
Davies-Jones: An efficient and accurate method for computing the
wet-bulb temperature along pseudoadiabats. Monthly Weather Review
(2008) vol. 136 (7) pp. 2764-2785
INPUTS:
T_k temperature (K)
p_t surface atmospheric pressure (pa)
T_k and p_t should be arrays of identical dimensions.
OUTPUTS:
es_mb vapor pressure (pa)
rs humidity (kg/kg)
de_mbdT d(es)/d(T)
dlnes_mbdT dln(es)/d(T)
rsdT d(qs)/d(T)
foftk Davies-Jones eqn 2.3
fdT d(f)/d(T)
Ported from HumanIndexMod by Jonathan R Buzan 08/08/13
MATLAB port by Robert Kopp
Last updated by Robert Kopp, robert-dot-kopp-at-rutgers-dot-edu, Wed Sep 02 22:22:25 EDT 2015
"""
# SHR_CONST_TKFRZ = 273.15;
lambd_a = 3.504 # Inverse of Heat Capacity
alpha = 17.67 # Constant to calculate vapour pressure
beta = 243.5 # Constant to calculate vapour pressure
epsilon = 0.6220 # Conversion between pressure/mixing ratio
es_C = 6.112 # Vapour Pressure at Freezing STD (mb)
vkp = 0.2854 # Heat Capacity
y0 = 3036 # constant
y1 = 1.78 # constant
y2 = 0.448 # constant
Cf = SHR_CONST_TKFRZ # Freezing Temp (K)
refpres = 1000 # Reference Pressure (mb)
# $$$ p_tmb % Pressure (mb)
# $$$ ndimpress % Non-dimensional Pressure
# $$$ prersdt % Place Holder for derivative humidity
# $$$ pminuse % Vapor Pressure Difference (mb)
# $$$ tcfbdiff % Temp diff ref (C)
# $$$ p0ndplam % dimensionless pressure modified by ref pressure
# $$$
# $$$ rsy2rs2 % Constant function of humidity
# $$$ oty2rs % Constant function of humidity
# $$$ y0tky1 % Constant function of Temp
# $$$
# $$$ d2e_mbdT2 % d2(es)/d(T)2
# $$$ d2rsdT2 % d2(r)/d(T)2
# $$$ goftk % g(T) exponential in f(T)
# $$$ gdT % d(g)/d(T)
# $$$ d2gdT2 % d2(g)/d(T)2
# $$$
# $$$ d2fdT2 % d2(f)/d(T)2 (K)
#
#-----------------------------------------------------------------------
# Constants used to calculate es(T)
# Clausius-Clapeyron
p_tmb = p_t*0.01
tcfbdiff = T_k - Cf + beta
es_mb = es_C * np.exp(alpha*(T_k - Cf)/(tcfbdiff))
dlnes_mbdT = alpha * beta/((tcfbdiff)*(tcfbdiff))
pminuse = p_tmb - es_mb
de_mbdT = es_mb * dlnes_mbdT
d2e_mbdT2 = dlnes_mbdT * (de_mbdT - 2*es_mb/(tcfbdiff))
# Constants used to calculate rs(T)
ndimpress = (p_tmb/refpres)**vkp
p0ndplam = refpres * ndimpress**lambd_a
rs = epsilon * es_mb/(p0ndplam - es_mb + np.spacing(1)) #eps)
prersdt = epsilon * p_tmb/((pminuse)*(pminuse))
rsdT = prersdt * de_mbdT
d2rsdT2 = prersdt * (d2e_mbdT2 -de_mbdT*de_mbdT*(2/(pminuse)))
# Constants used to calculate g(T)
rsy2rs2 = rs + y2*rs*rs
oty2rs = 1 + 2.0*y2*rs
y0tky1 = y0/T_k - y1
goftk = y0tky1 * (rs + y2 * rs * rs)
gdT = - y0 * (rsy2rs2)/(T_k*T_k) + (y0tky1)*(oty2rs)*rsdT
d2gdT2 = 2.0*y0*rsy2rs2/(T_k*T_k*T_k) - 2.0*y0*rsy2rs2*(oty2rs)*rsdT + \
y0tky1*2.0*y2*rsdT*rsdT + y0tky1*oty2rs*d2rsdT2
# Calculations for used to calculate f(T,ndimpress)
#print('Cf/T_k = '+str(Cf/T_k)+', '+str(lambd_a))
#print('vkp*lambd_a = '+ str(vkp)+', '+str(lambd_a))
#print('1-es_mb/p0ndplam = '+str(1 - es_mb/p0ndplam))
#exit()
foftk = ((Cf/T_k)**lambd_a)*(np.abs(1 - es_mb/p0ndplam))**(vkp*lambd_a)* \
np.exp(-lambd_a*goftk)
fdT = -lambd_a*(1.0/T_k + vkp*de_mbdT/pminuse + gdT)
d2fdT2 = lambd_a*(1.0/(T_k*T_k) - vkp*de_mbdT*de_mbdT/(pminuse*pminuse) - \
vkp*d2e_mbdT2/pminuse - d2gdT2)
# avoid bad numbers
rs[rs>1]=np.nan
rs[rs<0]=np.nan
return es_mb,rs,de_mbdT,dlnes_mbdT,rsdT,foftk,fdT
#end
def WetBulb(TemperatureC,Pressure,Humidity,HumidityMode=0):
"""
INPUTS:
TemperatureC 2-m air temperature (degrees Celsius)
Pressure Atmospheric Pressure (Pa)
Humidity Humidity -- meaning depends on HumidityMode
HumidityMode
0 (Default): Humidity is specific humidity (kg/kg)
1: Humidity is relative humidity (#, max = 100)
TemperatureC, Pressure, and Humidity should either be scalars or arrays of
identical dimension.
OUTPUTS:
Twb wet bulb temperature (C)
Teq Equivalent Temperature (K)
epott Equivalent Potential Temperature (K)
"""
# SHR_CONST_TKFRZ = 273.15
TemperatureK = TemperatureC + SHR_CONST_TKFRZ
constA = 2675 # Constant used for extreme cold temparatures (K)
grms = 1000 # Gram per Kilogram (g/kg)
p0 = 1000 # surface pressure (mb)
kappad = 0.2854 # Heat Capacity
C = SHR_CONST_TKFRZ # Freezing Temperature
pmb = Pressure*0.01 # pa to mb
T1 = TemperatureK # Use holder for T
es_mb,rs = QSat_2(TemperatureK, Pressure)[0:2] # first two returned values
if HumidityMode==0:
qin = Humidity # specific humidity
relhum = 100.0 * qin/rs # relative humidity (%)
vapemb = es_mb * relhum * 0.01 # vapor pressure (mb)
elif HumidityMode==1:
relhum = Humidity # relative humidity (%)
qin = rs * relhum * 0.01 # specific humidity
vapemb = es_mb * relhum * 0.01 # vapor pressure (mb)
#end
mixr = qin * grms # change specific humidity to mixing ratio (g/kg)
# real(r8) :: k1; % Quadratic Parameter (C)
# real(r8) :: k2; % Quadratic Parameter scaled by X (C)
# real(r8) :: pmb; % Atmospheric Surface Pressure (mb)
# real(r8) :: D; % Linear Interpolation of X
# real(r8) :: hot % Dimensionless Quantity used for changing temperature regimes
# real(r8) :: cold % Dimensionless Quantity used for changing temperature regimes
# real(r8) :: T1 % Temperature (K)
# real(r8) :: vapemb % Vapour Pressure (mb)
# real(r8) :: mixr % Mixing Ratio (g/kg)
# real(r8) :: es_mb_teq % saturated vapour pressure for wrt TEQ (mb)
# real(r8) :: de_mbdTeq % Derivative of Saturated Vapour pressure wrt TEQ (mb/K)
# real(r8) :: dlnes_mbdTeq % Log derivative of the sat. vap pressure wrt TEQ (mb/K)
# real(r8) :: rs_teq % Mixing Ratio wrt TEQ (kg/kg)
# real(r8) :: rsdTeq % Derivative of Mixing Ratio wrt TEQ (kg/kg/K)
# real(r8) :: foftk_teq % Function of EPT wrt TEQ
# real(r8) :: fdTeq % Derivative of Function of EPT wrt TEQ
# real(r8) :: wb_temp % Wet Bulb Temperature First Guess (C)
# real(r8) :: es_mb_wb_temp % Vapour Pressure wrt Wet Bulb Temp (mb)
# real(r8) :: de_mbdwb_temp % Derivative of Sat. Vapour Pressure wrt WB Temp (mb/K)
# real(r8) :: dlnes_mbdwb_temp % Log Derivative of sat. vap. pressure wrt WB Temp (mb/K)
# real(r8) :: rs_wb_temp % Mixing Ratio wrt WB Temp (kg/kg)
# real(r8) :: rsdwb_temp % Derivative of Mixing Ratio wrt WB Temp (kg/kg/K)
# real(r8) :: foftk_wb_temp % Function of EPT wrt WB Temp
# real(r8) :: fdwb_temp % Derivative of function of EPT wrt WB Temp
# real(r8) :: tl % Lifting Condensation Temperature (K)
# real(r8) :: theta_dl % Moist Potential Temperature (K)
# real(r8) :: pnd % Non dimensional Pressure
# real(r8) :: X % Ratio of equivalent temperature to freezing scaled by Heat Capacity
#-----------------------------------------------------------------------
# Calculate Equivalent Pot. Temp (pmb, T, mixing ratio (g/kg), pott, epott)
# Calculate Parameters for Wet Bulb Temp (epott, pmb)
pnd = (pmb/p0)**(kappad)
D = 1.0/(0.1859*pmb/p0 + 0.6512)
k1 = -38.5*pnd*pnd + 137.81*pnd - 53.737
k2 = -4.392*pnd*pnd + 56.831*pnd - 0.384
# Calculate lifting condensation level. first eqn
# uses vapor pressure (mb)
# 2nd eqn uses relative humidity.
# first equation: Bolton 1980 Eqn 21.
# tl = (2840/(3.5*log(T1) - log(vapemb) - 4.805)) + 55;
# second equation: Bolton 1980 Eqn 22. relhum = relative humidity
tl = (1.0/((1.0/((T1 - 55))) - (np.log(relhum/100.0)/2840.0))) + 55.0
# Theta_DL: Bolton 1980 Eqn 24.
theta_dl = T1*((p0/(pmb-vapemb))**kappad) * ((T1/tl)**(mixr*0.00028))
# EPT: Bolton 1980 Eqn 39.
epott = theta_dl * np.exp(((3.036/tl)-0.00178)*mixr*(1 + 0.000448*mixr))
Teq = epott*pnd # Equivalent Temperature at pressure
X = (C/Teq)**3.504
# Calculates the regime requirements of wet bulb equations.
invalid = (Teq > 600) + (Teq < 200)
hot = (Teq > 355.15)
cold = ((X>=1) * (X<=D))
X[invalid==1] = np.nan
Teq[invalid==1] = np.nan
# Calculate Wet Bulb Temperature, initial guess
# Extremely cold regime if X.gt.D then need to
# calculate dlnesTeqdTeq
es_mb_teq,rs_teq,de_mbdTeq, dlnes_mbdTeq, rsdTeq, foftk_teq, fdTeq = QSat_2(Teq, Pressure)
wb_temp = Teq - C - ((constA*rs_teq)/(1 + (constA*rs_teq*dlnes_mbdTeq)))
sub=np.where(X<=D)
wb_temp[sub] = (k1[sub] - 1.21 * cold[sub] - 1.45 * hot[sub] - (k2[sub] - 1.21 * cold[sub]) * X[sub] + (0.58 / X[sub]) * hot[sub])
wb_temp[invalid==1]=np.nan
# Newton-Raphson Method
maxiter = 3
iter = 0
delta = 1e6*np.ones_like(wb_temp)
while (np.max(delta)>0.01) and (iter<=maxiter):
es_mb_wb_temp,rs_wb_temp,de_mbdwb_temp, dlnes_mbdwb_temp, rsdwb_temp, foftk_wb_temp, fdwb_temp = QSat_2(wb_temp + C, Pressure)
delta = (foftk_wb_temp - X)/fdwb_temp #float((foftk_wb_temp - X)/fdwb_temp)
delta = np.where(delta<10., delta, 10.) #min(10,delta)
delta = np.where(delta>-10., delta, -10.) #max(-10,delta)
wb_temp = wb_temp - delta
wb_temp[invalid==1] = np.nan
Twb = wb_temp
iter = iter+1
#end
# ! 04-06-16: Adding iteration constraint. Commenting out original code.
# but in the MATLAB code, for sake of speed, we only do this for the values
# that didn't converge
if 1: #ConvergenceMode:
convergence = 0.00001
maxiter = 20000
es_mb_wb_temp,rs_wb_temp,de_mbdwb_temp, dlnes_mbdwb_temp, rsdwb_temp, foftk_wb_temp, fdwb_temp = QSat_2(wb_temp + C, Pressure)
delta = (foftk_wb_temp - X)/fdwb_temp #float((foftk_wb_temp - X)/fdwb_temp)
subdo = np.where(np.abs(delta)>convergence) #find(abs(delta)>convergence)
iter = 0
while (len(subdo[0])>0) and (iter<=maxiter):
iter = iter + 1
wb_temp[subdo] = wb_temp[subdo] - 0.1*delta[subdo]
es_mb_wb_temp,rs_wb_temp,de_mbdwb_temp, dlnes_mbdwb_temp, rsdwb_temp, foftk_wb_temp, fdwb_temp = QSat_2(wb_temp[subdo]+C, Pressure[subdo])
delta = 0 * wb_temp
delta[subdo] = (foftk_wb_temp - X[subdo])/fdwb_temp #float((foftk_wb_temp - X[subdo])/fdwb_temp)
subdo = np.where(np.abs(delta)>convergence) #find(abs(delta)>convergence);
#end
Twb = wb_temp
if any(map(len,subdo)): #len(subdo)>0:
print(len(subdo))
Twb[subdo] = TemperatureK[subdo]-C
#print(subdo)
for www in subdo[0]:
# print(www)
print('WARNING-Wet_Bulb failed to converge. Setting to T: WB, P, T, RH, Delta: %0.2f, %0.2f, %0.1f, %0.2g, %0.1f'%(Twb[www], Pressure[www], \
TemperatureK[www], relhum[www], delta[www]))
#end
#end
#end
#Twb=float(Twb)
return Twb,Teq,epott
if __name__ == "__main__":
tempC = np.array([31.,32.,33.,34.])
#tempd = np.array([26.58,29.10,29.26,27.55])
tempd = np.array([26.57,29.11,29.26,27.54])
RH = calc_RH_from_T_Td(tempC,tempd,mode=1)
print('RH = ')
print(RH)
#Pres = np.array([102130,102130,102130,102130])
Pres = np.array([101325]*4)
relHum = np.array([70.,80.,75.,60.])
Hum_mode = 1
Twb,Teq,epott = WetBulb(tempC,Pres,relHum,Hum_mode)
print(Twb)
print(Teq)
print(epott)