-
Notifications
You must be signed in to change notification settings - Fork 0
/
onsetdemise_stations_bimod.m
268 lines (241 loc) · 11.1 KB
/
onsetdemise_stations_bimod.m
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
% Matlab script to compute onset/demise (and duration)
% AG Munoz (IRI Columbia U) - [email protected] and Carlos Martinez - [email protected]
% Project: Predictability of the onset, duration and demise of the North and South American Monsoon Systems:
% the role of cross-equatorial interactions
% First edition: Aug 4, 2017
% Last edition: Feb 21, 2018
% Notes:
% + Subset of the full code written for the N/S American Monsoon
% project, to show the Bombardi and Carvalho approach.
% + Code reads rainfall data for indicated coordinates (1 gridbox) directly
% from the IRI Data Library (CPC-Unified - Chen et al 2008).
% + For illustration purposes, only one gridbox/station is used.
% + The method has issues when the year is too dry (as one should expect).
%%%%%START OF USER-MODIFIABLE SECTION%%%%%%%%%%%%
% 38 Stations in the Caribbean
% 57 Years of Data (1960 - 2017)
statn = 8 ;% Which station I want
sf=10; %smooth factor for the sgolay smoother
names = {'Antigua_Barbuda_IntlAP' ,'Nassau_IntlAP_Bahamas' ,'CIMH_Barbados', 'Gantley_Barbados', 'C_Farm_Belize', 'Intl_AP_Belize', 'Georgetown_Caymen'...
, 'Camaguey_Cuba', 'La_Habana_Cuba', 'DCAP_Dominica', 'Santo_Domingo_DR', 'Guadeloupe_IntlAP_Guadeloupe', 'Georgetown_Guyana', 'Time_HRI_Guyana' ...
, 'Worthy_Park_Jamaica', 'IntlAP_Martinique', 'Henaworra_IntlAP_St_Lucia', 'Dumbarton_St_Vincent', 'Corantjinpolder_Suriname', 'Zanderji_Suriname', 'Piarco_IntlAP_TT' ...
, 'Crown_Point_TT', 'Ft_Lauderdale_USA', 'Key_West_USA', 'Miami_IntlAP_USA', 'Palm_Beach_USA', 'Henry_E_R_IntilAP_St_Criox', 'Cyril_E_King_IntlAP_St_Thomas' ...
, 'Colosso_USPR', 'Dora_Bora_USPR', 'Ensenda_USPR', 'Guaynama_USPR', 'JaJome_Alto_USPR','Mora_Camp_USPR','Paraiso_USPR' ...
, 'Morovis_USPR', 'San_Andreas_Colombia', 'Felipe_Mexico'};
load dailydata.mat
%%%END OF USER-MODIFIABLE SECTION (DO NOT MODIFY ANYTHING BELOW THIS LINE)%%%%%
%%
%Daily and Pentad Record
%Reshape for Daily Data into 365 days for 57 years
for j = 1:38
yryrdaily.(names{j}) = reshape(dailydata(:,j), 365, []);
end
%Pentad Means
for i = 1:73
for k = 1:57
for j = 1:38
if nansum(isnan(yryrdaily.(names{j})(1+5*(i-1):5+5*(i-1),k))) > 2 %If pentad has 3 or more NaN's, then pentad is NaN
yrypentadmean.(names{j})(i,k) = NaN;
else
yrypentadmean.(names{j})(i,k) = nanmean(yryrdaily.(names{j})(1+5*(i-1):5+5*(i-1),k));
end
end
end
end
%Pentad Climatology
for i = 1:73
for j = 1:38
pentclim(i,j) = nanmean(yrypentadmean.(names{j})(i,:));
end
end
% pentclim replaces what was used as penmean
% pentad is replaced by the yrypentadmean
%% Angel M. Onset and Demise Code
% %Computing onset/demise a la Bombardi and Carvalho 2009:
clear onset demise inipent inipent2
pentad = yrypentadmean.(names{statn})(:,:); % Calls in the 29th Station (Coloso, PR) 73x57yrs
ns=size(pentad);nyear=ns(2);
%Find initial pentad for onset
display('Computing initial pentad... (this takes some time)')
%inipent/inipent2 is t0 in Bombardi and Carvalho 2009, initial pentad for
%onset and demise, respectively. Here wrt Jan 1 every year
clear f1 f2
if(~(isnan(pentclim(:,statn))))
%Compute first harmonic of the climatological rainfall for
%the location:
f1 = fit([1:73]',squeeze(pentclim(1:73,statn)),'fourier3');
yfitted = feval(f1,[1:73]);
[ypkon,idxon] = findpeaks(-yfitted);%min, for onset
[ypkde,idxde] = findpeaks(yfitted);%max, for demise
%Find x-coordinates of two larger maxima
[so,al]=sort(ypkde); %two maxima are at the end of so; al is location
%idxde(al(end))
%idxde(al(end-1))
%Find minimum before first large maximum, for rainfall onset
xfirstmax=min(idxde(al(end)),idxde(al(end-1)));
pp=find(idxon < xfirstmax);
inipent=idxon(pp(end)); %CHANGE FROM END-1 TO END
%Find minimum after second larg maximum, for rainfall demise
xsecmax=max(idxde(al(end)),idxde(al(end-1)));
%pp=find(idxon > xsecmax);
%inipent2=idxon(pp(1));
inipent2=xsecmax %MADE THIS INTO THE 2ND MAX FOR BEG. OF DEMISE
%Find minimum between two largest maxima, for MSD purposes
pp=find(idxon < xsecmax & idxon > xfirstmax);
inipent3=idxon(pp);
%----
%Older versions:
%inipent=round(fminbnd(f1,1,73)); %min, for onset
%inipent=round(interp1(feval(f1,[1:73]'),1:73,min(feval(f1,1:73))));%min, for onset
%f2 = fit([1:73]',squeeze(-penmean(1:73)),'fourier2'); %now negative for maximum
%inipent2=round(fminbnd(f2,1,73));%max, for demise
%inipent2=round(interp1(feval(f1,[1:73]'),1:73,max(feval(f1,1:73))));%max, for demise
%----
else
inipent=NaN;
inipent2=NaN;
inipent3=NaN;
end
inipent(inipent==0)=NaN;
inipent2(inipent2==0)=NaN;
display('t0 Done!')
figure(1);clf %Changed mean to nanmean for (pentad,2)
plot(f1);hold on; plot(1:73,nanmean(pentad,2))%,inipent,f1(inipent),'*k',inipent2,f1(inipent2),'*k','markers',12)
findpeaks(yfitted)
findpeaks(-yfitted)
%%
%Compute S
clear S S0
display('Computing S...')
for ity=1:nyear-1
S0=0;
S02=0;
%if(isnan(penmean) | isnan(inipent) | isnan(inipent2))
% S(ity)=NaN;
% S2(ity)=NaN; %S2 for demise, using inipent2
%else
for itp=1:73 %counter for
itk=inipent+itp;
itk2=inipent2+itp; %for demise, using inipent2
if itp == 1
S(ity,itp)=squeeze(pentad(inipent,ity)-pentclim(itp,statn));
S2(ity,itp)=pentad(inipent2,ity)-pentclim(itp,statn);
else
if itk<=73
S(ity,itp)=nansum(pentad(inipent:itk,ity)-pentclim(itp,statn),1);
else %use next year's data
S0=nansum(pentad(inipent:73,ity)-pentclim(itp,statn),1); %accumulated until the end of previous year
S(ity,itp)= S0 + ( nansum(pentad(1:itk-73,ity+1)-pentclim(itp,statn),1) ); %total, considering demise on following year
end
if itk2<=73 %demise
S2(ity,itp)=nansum(pentad(inipent2:itk2,ity)-pentclim(itp,statn),1);
else %use next year's data
S02=nansum(pentad(inipent2:73,ity)-pentclim(itp,statn),1); %accumulated until the end of previous year
S2(ity,itp)= S02 + ( nansum(pentad(1:itk2-73,ity+1)-pentclim(itp,statn),1) ); %total, considering demise on following year
end
end
end
%end
end
S(S==0)=NaN;
S2(S2==0)=NaN;
display('S Done!')
%Compute Sb
clear Sb
display('Smoothing with a Savitzky-Golay 2nd-degree filter...')
for ity=1:nyear-1
if( isnan(S(ity,:)) | isnan(S2(ity,:)) )
Sb(ity,:)=NaN;
Sb2(ity,:)=NaN;
else
%for pentads, R. Bombardi suggests to smooth S ~20 times;
%for daily data ~150 times !!! NOT USING THAT, USING SGOLAY
Sb(ity,1:73)=smooth(S(ity,1:73),sf,'sgolay');
Sb2(ity,1:73)=smooth(S2(ity,1:73),sf,'sgolay');
end
end
display('Sb Done!')
%Compute dSb/dt
clear dSb
%zci = @(v) find(v(:).*circshift(v(:), [-1 0]) <= 0); %function to find zero-crossings --NOT USED ANYMORE
for ity=1:nyear-1
if(isnan(Sb(ity,:)))
dSb(ity,:)=NaN;
else
dSb(ity,:)=diff(Sb(ity,:));
end
if(isnan(Sb2(ity,:)))
dSb2(ity,:)=NaN;
else
dSb2(ity,:)=diff(Sb2(ity,:));
end
end
display('dSb Done!')
%Compute onset and demise
clear onset demise
for ity=1:nyear-1
o=0;
d=0;
if(isnan(dSb(ity,:)) | isnan(dSb2(ity,:)))
onset(ity)=NaN;
demise(ity)=NaN;
else
for itp=2:71 %pentad relative to t0!!!
if o==1
else
if ( dSb(ity,itp-1) < 0 && dSb(ity,itp) > 0 && dSb(ity,itp+1) > 0 )
onset(ity)=itp+inipent; %relative to Jan 1st!
onset4anom=onset;
if onset(ity)>73
onset(ity)=onset(ity)-73;
end
o=1;
else
onset(ity)=NaN;
end
end
if d==1
else
if ( dSb2(ity,itp-1) > 0 && dSb2(ity,itp) < 0 && dSb2(ity,itp+1) < 0 )
demise(ity)=itp+inipent2; %relative to Jan 1st!
demise4anom=demise;
%if ( dSb(ilo,ila,ity,itp-1) > 0 && dSb(ilo,ila,ity,itp) < 0 && dSb(ilo,ila,ity,itp+1) < 0 )
%demise(ilo,ila,ity)=itp+inipent(ilo,ila); %relative to Jan 1st!
if demise(ity)>73
demise(ity)=demise(ity)-73;
end
d=1;
else
demise(ity)=NaN;
end
end
end
end
end
display('Onset and Demise Done!')
save -v7.3 onsetdemise.mat onset onset4anom demise demise4anom inipent inipent2 S Sb dSb pentad
%%
%TEST PLOTS
%Select a year:
iyr=30;
%
figure(2);clf %S,Sb,dSb for onset
pfig=plot(1:72,squeeze(S(iyr,1:72)),1:72,squeeze(Sb(iyr,1:72)),1:72,squeeze(dSb(iyr,1:72)),'LineWidth',3);hold on
bfig=bar(1:72,squeeze(pentad(1:72,iyr)')*10)
plot(1:73,pentad(:,iyr),onset(iyr),0,'*g',demise(iyr),0,'*k','markers',12,'LineWidth', 2)
title(names(statn))
ylabel('S,Sb,dSb for onset')
xlabel('pentad')
legend('S','Sb','dSb/dt','rainfallx10','zero line','onset','demise')
legend('Location','southwest')
legend('boxoff')
figure(3);clf %S,Sb,dSb for demise
pfig=plot(1:72,squeeze(S2(iyr,1:72)),1:72,squeeze(Sb2(iyr,1:72)),1:72,squeeze(dSb2(iyr,1:72)),'LineWidth',3);hold on
bfig=bar(1:72,squeeze(pentad(1:72,iyr)')*10)
plot(1:73,pentad(:,iyr),onset(iyr),0,'*g',demise(iyr),0,'*k','markers',15,'LineWidth', 2)
title(names(statn))
ylabel('S,Sb,dSb for demise')
xlabel('pentad')
legend('S','Sb','dSb/dt','rainfallx10','zero line','onset','demise')
legend('Location','southwest')
legend('boxoff')