Subversion Repositories public iLand

Rev

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