-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbadluk.f90
233 lines (226 loc) · 10.1 KB
/
badluk.f90
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
PROGRAM badluk
use moon_calc, only : debug_messages,ffrac,frac,hour,ifrac,im,iyyy&
&,jd,min,n,sec,timzon,frac_time_zone,julday,n_full_moons&
&,print_stats,flmoon,convert_time,qtime,piksr4
implicit none
INTEGER ic,icon,idwk,iybeg,iyend,jday,timezone,badcount,badmin,badmax,badtotal,whichbad,ntz,i,j,k ,l
LOGICAL newbad,rollback,check,list,dofrac,allbad
INTEGER, PARAMETER :: zs=-12,ze=14 ! The range of time zones to be searched.
character(len = 7) :: zn(zs:ze) ! Time zone name
character(len = 7) :: dzn(zs:ze), szn(zs:ze) ! Time zone name
character(len = 256) :: fmt
DATA iybeg,iyend /1900,2050/ ! The range of dates to be searched.
integer size
integer,allocatable :: bads(:,:) ! year,month,count
integer,allocatable :: times(:,:)
character(len = 3),allocatable :: stimes(:,:) ! string times
character(len = 5),allocatable :: sftimes(:,:) ! string fractional times
real,allocatable :: fyears(:)
! USES flmoon,julday
! compile options
rollback = .false. ! rollback to original output
check = .false. ! print check statements (debug)
debug_messages = check
list = .false. ! print dates as they are found
dofrac = .false. ! print factional hours (minutes)
! allocate array size
size=(iyend-iybeg)*12
allocate(bads(size,3),times(size,zs:ze),stimes(size,zs:ze),sftimes(size,zs:ze),fyears(size))
if(rollback) then
iybeg=1900
iyend=2000
check = .false.
endif
write (*,'(1x,a,i5,a,i5)') 'Full moons on Friday the 13th from',iybeg,' to',iyend
bads = 0
times = 0
stimes = ' :'
sftimes = ' :'
badmin = 0
badmax = 0
badtotal = 0
whichbad = 0
allbad = .false.
ntz=ze-zs+1
do timezone = zs, ze
! The full moon of Friday, June 13, 2014 did not (tecncially) occur in the Easter Time Zone
! (the program default). This loop is added to include other time zones.
if(check) write(*,'(a,sp,i4,a)')'testing zone',timezone,'...'
badcount = 0
write (zn(timezone), '(sp,i3,a)') timezone, ' UTC'
! zone names
zn( 0) = 'GMT'
if(rollback) zn(-5) = 'EST' ! Time zone −5 is Eastern Standard Time.
! standard time names
szn=''
szn(-8) = 'PST'
szn(-7) = 'MST'
szn(-6) = 'CST'
szn(-5) = 'EST'
! szn(-4) = 'AST'
! szn(+1) = 'CET'
! daylight saving time names
dzn=''
dzn(-7) = 'PDT'
dzn(-6) = 'MDT'
dzn(-5) = 'CDT'
dzn(-4) = 'EDT'
! dzn(-3) = 'ADT'
TIMZON=frac_time_zone(real(timezone))
do iyyy=iybeg,iyend ! Loop over each year,
do im=1,12 ! and each month.
jday=julday(im,13,iyyy) ! calculate the 13th
idwk=mod(jday+1,7) ! get weekday number
if(idwk.eq.5) then ! Is the 13th a Friday?
N=n_full_moons(iyyy,im)
! This value N is a first approximation to how many full moons have occurred since
! 1900. We will feed it into the phase routine and adjust it up or down until we
! determine that our desired 13th was or was not a full moon. The variable icon
! signals the direction of adjustment.
icon=0
1 call print_stats('LINE1',IYYY,IM,jday,JD,N,ic,icon)
call flmoon(N,2,JD,FRAC) ! Get date of full moon n.
call convert_time(IFRAC,FFRAC,FRAC,TIMZON,JD)
call qtime(FFRAC,HOUR,MIN,SEC)
if(JD.eq.jday)then ! Did we hit our target day?
if(badtotal.eq.0)then ! first?
if(check)write(*,'(1x,a,i2,a,i2,a,i4)')'found first bad day!',im,'/',13,'/',iyyy
badtotal = 1
whichbad = badtotal
else ! not first
if(check)write(*,'(a,i0.2,a,i0.2,a)',advance='no')' found a bad day at ',HOUR,':',MIN,'. checking...'
newbad = .true.
do i=1,badtotal ! check
if((iyyy.eq.bads(i,1)).and.(im.eq.bads(i,2))) then
whichbad = i
if(check)write(*,'(1x,i2,a,i2,a,i4,a,i2)')im,'/',13,'/',iyyy,' already found ',bads(i,3)
newbad = .false.
endif
enddo
if(newbad)then ! new?
badtotal = badtotal +1
whichbad = badtotal
if(badtotal.le.size)then
if(check) write(*,'(a,i2)')' found new bad day! count = ',badtotal
else
write(*,'(a,i3)') ' Too many bad days! Increase array size >',badtotal
stop
endif
endif ! end new?
endif ! end first?
bads(whichbad,1)=iyyy
bads(whichbad,2)=im
bads(whichbad,3)=bads(whichbad,3)+1
if(bads(whichbad,3)+1.eq.24)allbad=.true.
times(whichbad,timezone)=IFRAC
write(sftimes(whichbad,timezone),'(i0.2,a,i0.2)')HOUR,':',MIN
if((FFRAC.gt.1).and.(FFRAC.lt.1.5)) then
if(check) then
write(*,'(a,i4,a,i2,a,a5,a,a5)')' found margin: yr=',iyyy,' m=',im, &
' t0=',sftimes(whichbad,timezone) ,'; t=',sftimes(whichbad,timezone-1)
write(*,'(a,i0.2,a,i0.2,a,sp,i3)')' new time should be ',HOUR-1,':',MIN, ' in zone ',timezone-1
endif
write(sftimes(whichbad,timezone-1),'(i0.2,a,i0.2)')HOUR-1,':',MIN
endif
if( (rollback.and.(timezone.eq.-5).or.(.not.rollback).and.list.and.newbad) ) then
write (*,'(/1x,i2,a,i2,a,i4)') im,'/',13,'/',iyyy
if(len_trim(zn(timezone)).gt.0)then
write (fmt,'(a,i1,a)')'(1x,a,i2,a,a',len_trim(zn(timezone)),',a)'
else
write (fmt,*)'(1x,a,i2,a,a,a)'
endif
write (*,fmt) 'Full moon ',IFRAC,' hrs after midnight (',zn(timezone),')'
if((.not.rollback).and.dofrac) then
write (* ,'(1x,a,f4.1,a,a,a,i2,a,i2,a,i0.2,f0.1)')'Full moon ',FFRAC, &
' hrs after midnight (',zn(timezone),') at',HOUR,':',MIN,' :',int(SEC),SEC-real(int(SEC))
endif
endif
badcount = badcount + 1
! Don't worry if you are unfamiliar with FORTRAN's esoteric input /output
! statements; very few programs in this book do any input /output.
goto 2 ! Part of the break-structure, case of a match.
else ! Didn't hit it.
ic=isign(1,jday-JD)
call print_stats('CHECK',IYYY,IM,jday,JD,N,ic,icon)
if(ic.eq.-icon) goto 2 ! Another break, case of no match.
icon=ic
N=N+ic
endif
call print_stats('GOTO1',IYYY,IM,jday,JD,N,ic,icon)
goto 1
2 continue
endif
enddo
enddo
if(check) write (*,'(a,i2,a)') ' found ',badcount,' bad days in zone'
if(badcount.gt.badmax)then
badmax=badcount
endif
if(badcount.lt.badmax)then
badmin=badcount
endif
enddo
if(check) write(*,'(/a)') 'copying...'
write(fmt,'(a,i2,a)')'(1x,i2,1x,f6.1,i5,i3,',ntz,'(i3))'
do i=1,size
fyears(i)=real(bads(i,1))+real(bads(i,2))/12.
if((i.le.badtotal).and.check) write(*,fmt) i,fyears(i),bads(i,1),bads(i,2),times(i,:)
enddo
call piksr4(size,fyears,3,bads,ntz,times,5,sftimes)
k=size-badtotal ! calculate leading zeros in arrays
if(check) then
write(*,'(/a)') 'sorting...'
do i=1,size
l=i-k
if(i.gt.k) write(*,fmt) l,fyears(i),bads(i,1),bads(i,2),times(i,:)
enddo
endif
do i=1,size
do j=zs,ze
if(times(i,j).ne.0) write(stimes(i,j),'(i2)')times(i,j)
if((times(i,j).eq.0).and.(sftimes(i,j).ne.' :')) write(stimes(i,j),'(i2)')0
enddo
enddo
if(.not.rollback) then
write (*,'(/19x,a,i2,a,i5,a,i5)') 'Found ',badtotal ,' bad days from',iybeg,' to',iyend
write (*,'(1x,a,i2,a)') ' The luckiest zone had ',badmin ,' bad days'
write (*,'(1x,a,i2,a/)') 'The unluckiest zone had ',badmax ,' bad days'
if(.not.dofrac) then ! print hours only
write (fmt,'(a,i2,a)')'(a,',ntz,'(1x,a3))'
write (*,fmt) 'Daylight time ',(dzn(j),j=zs,ze) ! print daylight names
write (*,fmt) 'Standard time ',(szn(j),j=zs,ze) ! print standard names
write (*,fmt) 'UTC Offset ',(zn(j),j=zs,ze) ! print zone names
write (fmt,'(a,i2,a)')'(14x,',ntz,'(1x,sp,i3))'
write (fmt,'(a,i2,a)')'(1x,i2,1x,i2,a,i2,a,i4,',ntz ,'(1x,a3))'
write(*,*) repeat('-',int(13+4*ntz,8))
do, i=1,size
l=i-k
if(i.gt.k) then
write (*,fmt,advance='no') l,bads(i,2),'/',13,'/',bads(i,1),stimes(i,:)
if(bads(i,3).eq.24) write(*,'(a)',advance='no')' *'
write(*,*)
endif
enddo
write(*,*)repeat("-",int(13+4*ntz,8))
write(*,'(2x,a)' )'Note: times are rounded to the nearest hour.'
else ! print fractional hours
write (fmt,'(a,i2,a)')'(a,',ntz,'(3x,a3))'
write (*,fmt) 'Daylight time ',(dzn(j),j=zs,ze) ! print daylight names
write (*,fmt) 'Standard time ',(szn(j),j=zs,ze) ! print standard names
write (*,fmt) 'UTC Offset ',(zn(j),j=zs,ze) ! print zone names
write (fmt,'(a,i2,a)')'(1x,i2,1x,i2,a,i2,a,i4,',ntz ,'(1x,a5))'
write(*,*) repeat('-',int(13+6*ntz,8))
do, i=1,size
l=i-k
if(i.gt.k) then
write (*,fmt,advance='no') l,bads(i,2),'/',13,'/',bads(i,1),sftimes(i,:)
if(bads(i,3).eq.24) write(*,'(a)',advance='no')' *'
write(*,*)
endif
enddo
write(*,*)repeat("-",int(13+6*ntz,8))
write(*,'(2x,a)' )'Note: times do not include atmospheric refraction.'
endif
if(allbad) write(*,*) ' * World-wide bad luck'
endif
end program badluk