-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathDTC_GOT09.py
140 lines (100 loc) · 6.08 KB
/
DTC_GOT09.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
"""
example of GOT09 model
Ref:
Göttsche, F.M. and Olesen, F.S., 2009. Modelling the effect of optical thickness on diurnal cycles of land surface temperature. Remote Sensing of Environment, 113(11), pp.2306-2316.
https://www.sciencedirect.com/science/article/pii/S0034425709001850
"""
import numpy as np
import math
from matplotlib import pyplot as plt
from scipy.optimize import curve_fit
def GOT09(time, *param):
"""
Structure of the GOT09 model
:param time: time of the day, unit is hour
:param param:
:return:
"""
# parameter of the GOT09 model
# T0 is the residual temperature around sunrise
# Ta is the temperature amplitude
# tm is the time when temperature reaches its maximum
# ts is the time when free attenuation begins
# DeltaT (δT ) is the day-to-day change of residual temperature
# tao is total optical thickness
T0, Ta, tm, ts, DeltaT, tao = param[0], param[1], param[2], param[3],param[4],param[5]
theta = np.pi / 12 * (time - tm) # theta is the "thermal" hour angle with respect to thermal noon
theta_s = np.pi / 12 * (ts - tm) # theta_s is the "thermal" hour angle at the beginning of the free attenuation
mask_daytime = time < ts
mask_nighttime = time >= ts
Re, H = 6371, 8.43 # constant values of the Earth's radius and thickness of the atmosphere (“scale height”) in km
# cosine of the solar zenith angle
cos_sza = math.sin(delta / 180 * np.pi) * math.sin(latitude / 180 * np.pi) + math.cos(delta / 180 * np.pi) * math.cos(latitude / 180 * np.pi) * np.cos(theta)
# cosine of the solar zenith angle at the beginning of the free attenuation, i.e., theta = theta_s
cos_sza_s = math.sin(delta / 180 * np.pi) * math.sin(latitude / 180 * np.pi) + math.cos(delta / 180 * np.pi) * math.cos(latitude / 180 * np.pi) * math.cos(theta_s)
sin_sza_s = math.sqrt(1 - cos_sza_s * cos_sza_s)
# the minimum of the cosine of solor zenith angle, i.e., cos(theta) = 1 or time = tm
cos_sza_min = math.sin(delta / 180 * np.pi) * math.sin(latitude / 180 * np.pi) + math.cos(delta / 180 * np.pi) * math.cos(latitude / 180 * np.pi)
# the relative optical air mass
m_val = -Re / H * cos_sza + np.sqrt(pow((Re / H * cos_sza), 2) + 2 * Re / H + 1)
# the relative optical air mass at the beginning of the free attenuation
m_sza_s = -Re / H * cos_sza_s + math.sqrt(pow((Re / H * cos_sza_s), 2) + 2 * Re / H + 1)
# the relative optical air mass at the minimum of the cosine of solar zenith angle
m_min = -Re / H * cos_sza_min + math.sqrt(pow((Re / H * cos_sza_min), 2) + 2 * Re / H + 1)
# Eq.(21) in GOT09 paper
sza_derive_s = math.cos(delta / 180 * np.pi) * math.cos(latitude / 180 * np.pi) * math.sin(theta_s) / sin_sza_s
# Eq.(24) in GOT09 paper
m_derive_s = Re / H * sin_sza_s - pow(Re / H, 2) * cos_sza_s * sin_sza_s / math.sqrt(pow(Re / H * cos_sza_s, 2) + 2 * Re / H + 1)
k1 = 12 / np.pi / sza_derive_s # first part of k
k2 = tao * cos_sza_s * m_derive_s # second part of k
k3 = DeltaT / Ta * cos_sza_min / np.exp(tao*(m_min - m_sza_s)) # third part of k
# assemble three parts of k to calculate the final k value
k = k1 * (cos_sza_s - k3) / (sin_sza_s + k2)
# daytime temperature
temperature_day = T0 + Ta * cos_sza[mask_daytime] * np.exp(tao * (m_min - m_val[mask_daytime])) / cos_sza_min
temp1 = math.exp(tao * (m_min - m_sza_s)) / cos_sza_min
temp2 = np.exp(-12 / np.pi / k * (theta[mask_nighttime] - theta_s))
temperature_night = (T0 + DeltaT) + (Ta * cos_sza_s * temp1 - DeltaT) * temp2
temperature = np.concatenate((temperature_day, temperature_night))
return temperature
if __name__ == "__main__":
global latitude, delta
# Example data
latitude = 40.3581
longitude = -3.9803
doy = 139
# DTC model starts from the sunrise to the next sunrise, the values greater than 24 refer to the time in the next day
time_dtc = np.arange(4.75, 28.5, 1)
temperature_dtc = np.array(object=[283, 286, 290, 295, 301, 306, 309,
313, 314, 314, 312, 309, 305, 301,
297, 294, 293, 291, 290, 288, 287,
286, 285, 284],
dtype=float)
# delta is the solar declination
delta = 23.45 * np.sin(2 * np.pi / 365.0 * (284 + doy))
# omega is the duration of daytime
omega = 2.0 / 15 * math.acos(-math.tan(latitude / 180.0 * np.pi) * math.tan(delta / 180.0 * np.pi)) * 180.0 / np.pi
sunrise_time = 12 - omega / 2
sunset_time = 12 + omega / 2
# Setting the initial value
p0 = [np.average(temperature_dtc), np.max(temperature_dtc) - np.min(temperature_dtc), 13.0, sunset_time - 1, 0, 0.01]
# Solving the free parameters of GOT09 model. Bounds and max_nfev are optional
popt, pcov = curve_fit(GOT09, time_dtc, temperature_dtc, p0,
bounds=([200, 0, 8, 12, -5, 0], [400, 60, 17, 23, 5, 1]),
max_nfev = 10000)
print('fitted parameter', popt)
temperature_modeling = GOT09(time_dtc, popt[0], popt[1], popt[2], popt[3], popt[4], popt[5])
# calculate the root mean square error of the model fitting
print('RMSE of GOT09 model:', np.sqrt(np.mean(np.square(temperature_dtc - temperature_modeling))))
# plot the fitted curve
figure, ax = plt.subplots(nrows=1, ncols=1, figsize=(12, 6))
plt.title('Example of GOT09 model')
plt.plot(time_dtc, temperature_dtc, '.g', label='LST observations')
plt.plot(time_dtc, temperature_modeling, 'r', label='DTC modelling results')
ax.tick_params('x', labelsize=12, direction='out', length=4, width=1.5, bottom=True, which='major')
ax.tick_params('y', labelsize=12, direction='out', length=4, width=1.5, left=True, which='major')
ax.set_xlabel('Time (h)', size=15)
ax.set_ylabel('LST', size=15)
plt.title('Example of GOT09 model', fontsize=18)
plt.legend(loc='best')
plt.show()