@@ -23,11 +23,11 @@ DAFunctionWallHeatFlux::DAFunctionWallHeatFlux(
23
23
const DAIndex & daIndex ,
24
24
const word functionName )
25
25
: DAFunction (
26
- mesh ,
27
- daOption ,
28
- daModel ,
29
- daIndex ,
30
- functionName ),
26
+ mesh ,
27
+ daOption ,
28
+ daModel ,
29
+ daIndex ,
30
+ functionName ),
31
31
wallHeatFlux_ (
32
32
IOobject (
33
33
"wallHeatFlux ",
@@ -40,10 +40,15 @@ DAFunctionWallHeatFlux::DAFunctionWallHeatFlux(
40
40
" calculated ")
41
41
{
42
42
43
+ // check and assign values for scheme and formulation
44
+ formMode_ = functionDict_ .lookupOrDefault < word > ("formulation" , "default" );
45
+ calcMode_ = functionDict_ .lookupOrDefault < bool > ("byUnitArea" , true);
46
+
43
47
if (mesh_ .thisDb ().foundObject < DATurbulenceModel > ("DATurbulenceModel" ))
44
48
{
45
49
DATurbulenceModel & daTurbModel =
46
50
const_cast < DATurbulenceModel & > (daModel_ .getDATurbulenceModel ());
51
+
47
52
if (daTurbModel .getTurbModelType () == "incompressible" )
48
53
{
49
54
// initialize the Prandtl number from transportProperties
@@ -89,7 +94,7 @@ DAFunctionWallHeatFlux::DAFunctionWallHeatFlux(
89
94
}
90
95
}
91
96
92
- /// calculate the value of objective function
97
+ //---------- Calculate Objective Function Value ----------
93
98
scalar DAFunctionWallHeatFlux ::calcFunction ()
94
99
{
95
100
/*
@@ -101,85 +106,197 @@ scalar DAFunctionWallHeatFlux::calcFunction()
101
106
functionValue: the sum of objective, reduced across all processors and scaled by "scale"
102
107
*/
103
108
104
- // always calculate the area of all the heat flux patches
105
- areaSum_ = 0.0 ;
106
- forAll (faceSources_ , idxI )
109
+ // only calculate the area of all the heat flux patches if scheme is byUnitArea
110
+ if (calcMode_ )
107
111
{
108
- const label & functionFaceI = faceSources_ [idxI ];
109
- label bFaceI = functionFaceI - daIndex_ .nLocalInternalFaces ;
110
- const label patchI = daIndex_ .bFacePatchI [bFaceI ];
111
- const label faceI = daIndex_ .bFaceFaceI [bFaceI ];
112
- areaSum_ += mesh_ .magSf ().boundaryField ()[patchI ][faceI ];
112
+ areaSum_ = 0.0 ;
113
+ forAll (faceSources_ , idxI )
114
+ {
115
+ const label & functionFaceI = faceSources_ [idxI ];
116
+ label bFaceI = functionFaceI - daIndex_ .nLocalInternalFaces ;
117
+ const label patchI = daIndex_ .bFacePatchI [bFaceI ];
118
+ const label faceI = daIndex_ .bFaceFaceI [bFaceI ];
119
+ areaSum_ += mesh_ .magSf ().boundaryField ()[patchI ][faceI ];
120
+ }
121
+ reduce (areaSum_ , sumOp < scalar > ( ));
113
122
}
114
- reduce (areaSum_ , sumOp < scalar > ( ));
115
123
116
124
// initialize objFunValue
117
125
scalar functionValue = 0.0 ;
118
126
119
127
volScalarField ::Boundary & wallHeatFluxBf = wallHeatFlux_ .boundaryFieldRef ();
120
128
129
+ // calculate HFX for fluid domain
121
130
if (mesh_ .thisDb ().foundObject < DATurbulenceModel > ("DATurbulenceModel" ))
122
131
{
123
132
DATurbulenceModel & daTurbModel =
124
133
const_cast < DATurbulenceModel & > (daModel_ .getDATurbulenceModel ());
134
+
135
+ // calculate HFX for incompressible flow (no he for incompressible -> HFX = Cp * alphaEff * dT/dz)
125
136
if (daTurbModel .getTurbModelType () == "incompressible" )
126
137
{
127
- // incompressible flow does not have he, so we do H = Cp * alphaEff * dT/dz
128
138
const objectRegistry & db = mesh_ .thisDb ();
129
139
const volScalarField & T = db .lookupObject < volScalarField > ("T" );
130
140
volScalarField alphaEff = daTurbModel .alphaEff ();
131
141
const volScalarField ::Boundary & TBf = T .boundaryField ();
132
142
const volScalarField ::Boundary & alphaEffBf = alphaEff .boundaryField ();
143
+
133
144
forAll (wallHeatFluxBf , patchI )
134
145
{
135
146
if (!wallHeatFluxBf [patchI ].coupled ())
136
147
{
137
- wallHeatFluxBf [patchI ] = Cp_ * alphaEffBf [patchI ] * TBf [patchI ].snGrad ();
148
+ // use OpenFOAM's snGrad()
149
+ if (formMode_ == "default" )
150
+ {
151
+ wallHeatFluxBf [patchI ] = Cp_ * alphaEffBf [patchI ] * TBf [patchI ].snGrad ();
152
+ }
153
+ // use DAFOAM's custom formulation
154
+ else if (formMode_ == "daCustom" )
155
+ {
156
+ forAll (wallHeatFluxBf [patchI ], faceI )
157
+ {
158
+ scalar T2 = TBf [patchI ][faceI ];
159
+ label nearWallCellIndex = mesh_ .boundaryMesh ()[patchI ].faceCells ()[faceI ];
160
+ scalar T1 = T [nearWallCellIndex ];
161
+ vector c1 = mesh_ .Cf ().boundaryField ()[patchI ][faceI ];
162
+ vector c2 = mesh_ .C ()[nearWallCellIndex ];
163
+ scalar d = mag (c1 - c2 );
164
+ scalar dTdz = (T2 - T1 ) / d ;
165
+ wallHeatFluxBf [patchI ][faceI ] = Cp_ * alphaEffBf [patchI ][faceI ] * dTdz ;
166
+ }
167
+ }
168
+ // error message incase of invalid entry
169
+ else
170
+ {
171
+ FatalErrorIn (" " ) << "formulation: "
172
+ << formMode_ << " not supported!"
173
+ << " Options are: default and daCustom."
174
+ << abort (FatalError );
175
+ }
138
176
}
139
177
}
140
178
}
179
+ // calculate HFX for compressible flow (HFX = alphaEff * dHe/dz)
141
180
else if (daTurbModel .getTurbModelType () == "compressible" )
142
181
{
143
- // compressible flow, H = alphaEff * dHE/dz
144
182
fluidThermo & thermo = const_cast < fluidThermo & > (
145
183
mesh_ .thisDb ().lookupObject < fluidThermo > ("thermophysicalProperties" ));
146
184
volScalarField & he = thermo .he ();
147
185
const volScalarField ::Boundary & heBf = he .boundaryField ();
148
186
volScalarField alphaEff = daTurbModel .alphaEff ();
149
187
const volScalarField ::Boundary & alphaEffBf = alphaEff .boundaryField ();
188
+
150
189
forAll (wallHeatFluxBf , patchI )
151
190
{
152
191
if (!wallHeatFluxBf [patchI ].coupled ())
153
192
{
154
- wallHeatFluxBf [patchI ] = alphaEffBf [patchI ] * heBf [patchI ].snGrad ();
193
+ // use OpenFOAM's snGrad()
194
+ if (formMode_ == "default" )
195
+ {
196
+ wallHeatFluxBf [patchI ] = alphaEffBf [patchI ] * heBf [patchI ].snGrad ();
197
+ }
198
+ // use DAFOAM's custom formulation
199
+ else if (formMode_ == "daCustom" )
200
+ {
201
+ forAll (wallHeatFluxBf [patchI ], faceI )
202
+ {
203
+ scalar He2 = heBf [patchI ][faceI ];
204
+ label nearWallCellIndex = mesh_ .boundaryMesh ()[patchI ].faceCells ()[faceI ];
205
+ scalar He1 = he [nearWallCellIndex ];
206
+ vector c1 = mesh_ .Cf ().boundaryField ()[patchI ][faceI ];
207
+ vector c2 = mesh_ .C ()[nearWallCellIndex ];
208
+ scalar d = mag (c1 - c2 );
209
+ scalar dHedz = (He2 - He1 ) / d ;
210
+ wallHeatFluxBf [patchI ][faceI ] = alphaEffBf [patchI ][faceI ] * dHedz ;
211
+ }
212
+ }
213
+ // error message incase of invalid entry
214
+ else
215
+ {
216
+ FatalErrorIn (" " ) << "formulation: "
217
+ << formMode_ << " not supported!"
218
+ << " Options are: default and daCustom."
219
+ << abort (FatalError );
220
+ }
155
221
}
156
222
}
157
223
}
158
224
}
225
+
226
+ // calculate HFX for solid domain (HFX = k * dT/dz, where k = DT / rho / Cp)
159
227
else
160
228
{
161
- // solid. H = k * dT/dz, where k = DT / rho / Cp
162
229
const objectRegistry & db = mesh_ .thisDb ();
163
230
const volScalarField & T = db .lookupObject < volScalarField > ("T" );
164
231
const volScalarField ::Boundary & TBf = T .boundaryField ();
232
+
165
233
forAll (wallHeatFluxBf , patchI )
166
234
{
167
235
if (!wallHeatFluxBf [patchI ].coupled ())
168
236
{
169
- wallHeatFluxBf [patchI ] = k_ * TBf [patchI ].snGrad ();
237
+
238
+ // use OpenFOAM's snGrad()
239
+ if (formMode_ == "default" )
240
+ {
241
+ wallHeatFluxBf [patchI ] = k_ * TBf [patchI ].snGrad ();
242
+ }
243
+ // use DAFOAM's custom formulation
244
+ else if (formMode_ == "daCustom" )
245
+ {
246
+ forAll (wallHeatFluxBf [patchI ], faceI )
247
+ {
248
+ scalar T2 = TBf [patchI ][faceI ];
249
+ label nearWallCellIndex = mesh_ .boundaryMesh ()[patchI ].faceCells ()[faceI ];
250
+ scalar T1 = T [nearWallCellIndex ];
251
+ vector c1 = mesh_ .Cf ().boundaryField ()[patchI ][faceI ];
252
+ vector c2 = mesh_ .C ()[nearWallCellIndex ];
253
+ scalar d = mag (c1 - c2 );
254
+ scalar dTdz = (T2 - T1 ) / d ;
255
+ wallHeatFluxBf [patchI ][faceI ] = k_ * dTdz ;
256
+ }
257
+ }
258
+ // error message incase of invalid entry
259
+ else
260
+ {
261
+ FatalErrorIn (" " ) << "formulation: "
262
+ << formMode_ << " not supported!"
263
+ << " Options are: default and daCustom."
264
+ << abort (FatalError );
265
+ }
170
266
}
171
267
}
172
268
}
269
+
173
270
// calculate area weighted heat flux
271
+ scalar val = 0 ;
174
272
forAll (faceSources_ , idxI )
175
273
{
176
274
const label & functionFaceI = faceSources_ [idxI ];
177
275
label bFaceI = functionFaceI - daIndex_ .nLocalInternalFaces ;
178
276
const label patchI = daIndex_ .bFacePatchI [bFaceI ];
179
277
const label faceI = daIndex_ .bFaceFaceI [bFaceI ];
180
-
181
278
scalar area = mesh_ .magSf ().boundaryField ()[patchI ][faceI ];
182
- scalar val = scale_ * wallHeatFluxBf [patchI ][faceI ] * area / areaSum_ ;
279
+
280
+ // represent wallHeatFlux by unit area
281
+ if (calcMode_ )
282
+ {
283
+ val = scale_ * wallHeatFluxBf [patchI ][faceI ] * area / areaSum_ ;
284
+ }
285
+ // represent wallHeatFlux as total heat transfer through surface
286
+ else if (!calcMode_ )
287
+ {
288
+ val = scale_ * wallHeatFluxBf [patchI ][faceI ] * area ;
289
+ }
290
+ // error message incase of invalid entry
291
+ else
292
+ {
293
+ FatalErrorIn (" " ) << "byUnitArea: "
294
+ << calcMode_ << " not supported!"
295
+ << " Options are: True (default) and False."
296
+ << abort (FatalError );
297
+ }
298
+
299
+ // update obj. func. val
183
300
functionValue += val ;
184
301
}
185
302
0 commit comments