Rev 1221 | Details | Compare with Previous | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
15 | Werner | 1 | |
2 | #include "solarradiation.h" |
||
3 | |||
4 | #include "hemigrid.h" |
||
5 | |||
6 | |||
7 | //#include "globals.h" |
||
8 | //#include "modell.h" |
||
9 | //#include "DBUtil.h" |
||
10 | |||
11 | |||
12 | //--------------------- SolarRad ------------- |
||
13 | // ------ Mini-Modul zur Berechnung der Globalstrahlung |
||
14 | // eigentlich ähnlich wie Strahlungsmodell Picus 1.x, |
||
15 | // jedoch gleich selber machen ist einfacher.... |
||
16 | void SolarRadiation::Setup(double BreiteGrad) |
||
17 | { |
||
18 | Latitude=BreiteGrad * M_PI / 180.; |
||
19 | for (int day=0; day<366; day++) { |
||
20 | calcDay(day); |
||
21 | } |
||
22 | } |
||
23 | |||
24 | |||
25 | |||
26 | ////////////////////////////////////////////////////// |
||
27 | /// Calculate factor for optical mass (used for radiation calculation) |
||
28 | /// http://www.sortie-nd.org/help/manuals/developer_documentation/cplusplus/light.html |
||
29 | ////////////////////////////////////////////////////// |
||
30 | double GetOpticalMass(const double elevation_rad) |
||
31 | { |
||
32 | if (elevation_rad < 5*M_PI/180.) |
||
33 | return 10.39; |
||
34 | |||
35 | if (elevation_rad < 15*M_PI/180.) |
||
36 | return 5.6; |
||
37 | |||
38 | if (elevation_rad < 25*M_PI/180.) |
||
39 | return 2.9; |
||
40 | |||
41 | // if >= 25°: |
||
42 | // cos(zenith) == cos(90° - elevation).... |
||
43 | // here i am not completely sure: but sec(x)=1/cos(x) |
||
44 | // seems to be right. sec(25°)=2.3, sec(20)=2.9 (see above!), sec(0)=1 |
||
45 | // the concrete steps above (5,15,25°) can be observed in cumulated outputs |
||
46 | // e.g. global radiation over elevation - maybe a smoother variant should be used. |
||
47 | |||
48 | return 1./cos(M_PI/2. - elevation_rad); |
||
49 | } |
||
50 | ////////////////////////////////////////////////////// |
||
51 | // calc relative radiation matrix. |
||
52 | ////////////////////////////////////////////////////// |
||
53 | void SolarRadiation::calculateRadMatrix(const float Step_deg, HemiGrid &Grid) |
||
54 | { |
||
55 | |||
56 | // const double transmissionskoeff[12]={ |
||
57 | // 0.4, 0.48, 0.47, 0.46, 0.47, 0.44, |
||
58 | // 0.48, 0.44, 0.39, 0.38, 0.32, 0.34}; |
||
59 | |||
60 | // elevation angle threshold |
||
61 | const double min_elevation_angle = 0*M_PI/180.; |
||
62 | double latidude = Latitude; |
||
63 | |||
64 | Grid.setup(Step_deg); |
||
24 | Werner | 65 | int cntAzimuth = Grid.matrixCountAzimuth(); |
66 | int cntElevation = Grid.matrixCountElevation(); |
||
15 | Werner | 67 | |
68 | int day; |
||
69 | double sin_declination, declination, eccentricity; |
||
70 | double current_transmission; |
||
71 | double sun_day, diff_hour; |
||
72 | double hour, sun_hour; |
||
73 | double step = 1./60.; // 1 minute |
||
74 | bool sun_rise, sun_down; |
||
75 | double total_rad_sum = 0; |
||
76 | double elevation, sin_elevation, cos_azimuth, azimuth; |
||
77 | double Iincoming; |
||
78 | for (day = 0; day < 366; day++) { |
||
79 | // skip days due to vegetation period limit: |
||
80 | if (day<mMinDay || day>mMaxDay) |
||
81 | continue; |
||
82 | sin_declination=0.39785*sin((278.97+0.9856*day+1.9165*sin((356.6+0.9856*day)*M_PI/180.))*M_PI/180.); |
||
83 | declination=asin(sin_declination); // declination of earth (rad) |
||
84 | // solarconstant: Gassel p. 15 |
||
85 | SolarConstant=1356.5+48.5*cos(0.01721*(day-15)); |
||
86 | sun_day = (day-1)*2.*M_PI/365.; // rad |
||
87 | // difference between "real sunhour" and hour |
||
88 | diff_hour = 12+0.12357*sin(sun_day)-0.004289*cos(sun_day)+0.153809*sin(2.*sun_day)+0.060783*cos(2.*sun_day); |
||
89 | // eccentricity: SORTIE (http://www.sortie-nd.org/help/manuals/developer_documentation/cplusplus/light.html) |
||
90 | eccentricity = 1.000110 + 0.034221* cos(sun_day) + 0.001280*sin(sun_day) + 0.000719 * cos(2*sun_day) + 0.000077* sin(2*sun_day); |
||
91 | hour = 0; |
||
92 | sun_rise = sun_down = false; |
||
93 | //current_transmission = transmissionskoeff[(day/30)%12]; |
||
94 | current_transmission = 0.75; |
||
95 | while (!sun_down) { |
||
96 | sun_hour=(hour-diff_hour)*15./180*M_PI; // s in rad |
||
97 | // sin elevation: =sin(lat)*sin(decl) + cos(lat)*cos(sunhour)*cos(decl) |
||
98 | sin_elevation=sin(latidude)*sin_declination + cos(latidude)*cos(sun_hour)*cos(declination); |
||
99 | elevation =asin(sin_elevation); // Sonnenhöhe [rad] |
||
100 | // check for state of the sun |
||
101 | if (!sun_rise && elevation>min_elevation_angle) |
||
102 | sun_rise=true; |
||
103 | if (sun_rise && elevation<min_elevation_angle) |
||
104 | sun_down=true; |
||
105 | if (elevation>=min_elevation_angle) { |
||
106 | // calculate azimuth |
||
107 | cos_azimuth = ( sin_elevation * sin(latidude) - sin_declination) / ( cos(elevation)*cos(latidude) ); |
||
108 | azimuth = acos(cos_azimuth); |
||
109 | // azimuth is always > 0: use sign of sun_hour: if negative, than direction is "east" (before noon) |
||
110 | if (sun_hour < 0.) |
||
111 | azimuth*=-1; |
||
112 | // calculate the energy (reduction due to transmissioncoeffiecients..) |
||
113 | // I = eccentricity * trans^optical_mass * cos(zeninthangle!) |
||
114 | Iincoming = eccentricity * pow(current_transmission, GetOpticalMass(elevation)) * cos(M_PI/2. - elevation); |
||
115 | total_rad_sum += Iincoming; |
||
116 | |||
117 | // azimuth is 0 for "south". the grid is north-oriented -> reverse (azimuth(North)=0) |
||
118 | if (azimuth <= 0.) |
||
119 | azimuth += M_PI; |
||
120 | else |
||
121 | azimuth -= M_PI; |
||
122 | |||
123 | // store in grid: |
||
124 | Grid.rGet(azimuth, elevation) += Iincoming; |
||
125 | |||
126 | } // elevation>min_elevation_angle |
||
127 | hour += step; |
||
128 | if (hour>24) |
||
129 | throw QString("tsolarrad::calcmatrix: elevation never above minimum elevation."); |
||
130 | } // while |
||
131 | } // for (each day) |
||
132 | |||
133 | // normalize the array and add the diffus radiation... |
||
134 | // direct: sum(all rad) should be fraction of direct radiation |
||
135 | double direct_rad_multiplier = (1.-mDiffusRadFraction) / ( total_rad_sum ); |
||
136 | // diffuse: the light is evenly distributed (above the minimum angle) |
||
137 | // add a little to lower bound to avoid rounding down |
||
138 | int min_elev_index = int( (min_elevation_angle+0.0001) / (M_PI / 2.) * cntElevation); |
||
139 | double diffuse_rad_addon = mDiffusRadFraction / ((cntElevation - min_elev_index) * cntAzimuth); |
||
140 | |||
141 | for (int i=min_elev_index; i<cntElevation; i++) { |
||
142 | for (int j=0;j<cntAzimuth;j++) { |
||
143 | double &cell_value = Grid.rGetByIndex(j, i); |
||
144 | cell_value*=direct_rad_multiplier; |
||
145 | cell_value+=diffuse_rad_addon; |
||
146 | } |
||
147 | } |
||
148 | |||
149 | } |
||
150 | |||
151 | |||
152 | void SolarRadiation::calcDay(int day) |
||
153 | { |
||
154 | // Berechnungen pro Tag: |
||
155 | // Deklination: |
||
156 | //sin D=0,39785*SIN((278,97+0,9856*G8+1,9165*SIN((356,6+0,9856*G8)*M_PI()/180))*M_PI()/180) |
||
157 | double sinDekl=0.39785*sin((278.97+0.9856*day+1.9165*sin((356.6+0.9856*day)*M_PI/180.))*M_PI/180.); |
||
158 | double Dekl=asin(sinDekl); // Deklination in Grad |
||
159 | //S = 1356,5+48,5*cos[0,01721*(n-15)] Gassel p. 15 |
||
160 | |||
161 | SolarConstant=1356.5+48.5*cos(0.01721*(day-15)); |
||
162 | // Sonnentag |
||
163 | double SunDay; |
||
164 | SunDay=(day-1)*2.*M_PI/365.; // [rad] |
||
165 | // Unterschied wahre Sonnenstunde - Stunde |
||
166 | double Diff=12+0.12357*sin(SunDay)-0.004289*cos(SunDay)+0.153809*sin(2.*SunDay)+0.060783*cos(2.*SunDay); |
||
167 | |||
168 | // Integration über den Tag |
||
169 | double Sum=0., Length=0., Rad; |
||
170 | // Sonnenstunde |
||
171 | double Hour, SunHour, sinH, Altitude; |
||
172 | bool SunRise=false; |
||
173 | bool SunDown=false; |
||
174 | Hour=0.; |
||
175 | const double Step=1/60.; // Minutentakt |
||
176 | // berechnung der tageslänge methode 2: |
||
177 | // tag: wenn strahlung > threshold |
||
178 | double RadThreshold = -1; // not used: Picus specific: =GModell->Settings["BBDayLengthMode"].AsDouble(); |
||
179 | const double transmissionskoeff[12]={ |
||
180 | 0.4, 0.48, 0.47, 0.46, 0.47, 0.44, |
||
181 | 0.48, 0.44, 0.39, 0.38, 0.32, 0.34}; |
||
182 | while (!SunDown) |
||
183 | { |
||
184 | SunHour=(Hour-Diff)*15./180*M_PI; // s in rad |
||
185 | // sin Sonnenhöhe: =sin(Breite)*sin(Dekl) + cos(Breite)*cos(Sonnenstunde)*cos(Deklination) |
||
186 | sinH=sin(Latitude)*sinDekl + cos(Latitude)*cos(SunHour)*cos(Dekl); |
||
187 | Altitude=asin(sinH); // Sonnenhöhe [rad] |
||
188 | // calculate azimuth-angle |
||
189 | if (!SunRise && Altitude>0) |
||
190 | SunRise=true; |
||
191 | if (SunRise && Altitude<0) |
||
192 | SunDown=true; |
||
193 | if (Altitude>0) { |
||
194 | // Length: Tageslänge [Stunden] |
||
195 | if (RadThreshold==-1) |
||
196 | Length+=Step; // default |
||
197 | else if (SolarConstant*sinH*transmissionskoeff[(day/30)%12] > RadThreshold) { |
||
198 | // SolarConst[W/m2]*frac*frac=[W/m2] |
||
199 | Length+=Step; // strahlungs-grenze-fall |
||
200 | } |
||
201 | // Rad: W * h * frac: Wh |
||
202 | Rad=SolarConstant * Step * sinH; |
||
203 | Sum+=Rad; |
||
204 | } |
||
205 | Hour+=Step; |
||
206 | } |
||
207 | RadExtraTerrestrisch[day]=Sum*3.6; // Wh-> kJ (*3600/1000) |
||
208 | DayLength[day]=Length*3600; // sekunden |
||
209 | } |
||
210 |