CFD Online Discussion Forums

CFD Online Discussion Forums (https://www.cfd-online.com/Forums/)
-   OpenFOAM Programming & Development (https://www.cfd-online.com/Forums/openfoam-programming-development/)
-   -   How to update a volScalarField's values (https://www.cfd-online.com/Forums/openfoam-programming-development/125308-how-update-volscalarfields-values.html)

ThomasV October 23, 2013 04:09

How to update a volScalarField's values
 
Hi!

I have a question concerning how to update a volScalarField. Imagine the following problem:

You have a volScalarField T for the temperature (with values applied to it). You now have another volScalarField X which is meant to consist of values depending on the respective temperatures. How do I implement an update for X which looks up the temperature of each element and then updates the value in terms of special conditions. With that I mean a procedure of this sort:

Code:

if (T<100)
{
    X = 1;
}

if (T=100)
{
    X = 2;
}

if (T>100)
{
    X = 3;
}


alexeym October 23, 2013 04:26

There are several ways to do it. One of them:

Code:

forAll(X, cellI) {
    if (T[cellI] < 100)
        X[cellI] = 1;
    else if (T[cellI] > 100)
        X[cellI] = 3;
    else
        X[cellI] = 2;
}


dkxls October 23, 2013 04:44

There are many examples in the source code, see e.g.:
https://github.com/OpenFOAM/OpenFOAM.../hePsiThermo.C

Here a simplified example:
Say you get your temperature field like this:
Code:

    const volScalarField& T = thermo.T();
Then you can update your field X like this:
Code:

    scalarField& TCells = T.internalField();

    forAll(TCells, celli)
    {
        if (T[celli] < 100.0)
            X[celli] = 1;
        else if (T[celli] > 100.0)
            X[celli] = 3;
        else
            X[celli] = 2;
    }

The boundary field you can update in a similar manner, I suggest you have a look at the code I linked above if you need to do this.

ThomasV October 23, 2013 05:13

Thanks! :)

EDIT:
Btw - is there a difference between writing "celli" and "cellI" (i.e. upper- / lowercase letter)?

dkxls October 23, 2013 08:15

Quote:

Originally Posted by ThomasV (Post 458459)
Btw - is there a difference between writing "celli" and "cellI" (i.e. upper- / lowercase letter)?

No, there is no difference. :)
The variable celli is a 'label' (i.e. a integer) defined in forAll, which is a macro that sets up the for loop. So, you can call the variable whatever you want, you just need to be consistent. ;)

ThomasV October 28, 2013 10:06

Allright thanks...

I'd like to add another more "practical" question here. Things now work fine and I define a temperature field which then is analyzed to determine the solid fraction in the respective cells. What I'd like to know is how to write the new volScalarField for the fraction solid into the "0 folder". I create the fraction solid field like this:
Code:

    volScalarField fracSol
    (
        IOobject
        (
            "fracSol",
            runTime.timeName(),
            mesh,
            IOobject::NO_READ,
            IOobject::AUTO_WRITE
        ),
        mesh,
        scalar(0)
    );

Afterwards it becomes filled with the actual data determined by the respective temperatures of the cells. This way I get my fracSol file written in every timestep. Every timestep except for the initial one that is...

I now would like to know how I also can write the calculated fracSol into the 0 folder i.e. the fraction solid field for the initial temperatures. This also would come in handy when postprocessing the project in ParaFoam as there now is no fracSol field in time step 0 and it is a bit tedious to go to timestep 1 and activate the field plus once again any time you accidentally rewind to timestep 0 as it then gets unloaded (because there's no data for it in step 0)...

I already tried to simply put a runtime.write() command before progressing time in the time loop but this didn't do the trick. I guess one usually would approach this problem with the setfields utility but as far as I know one here is limited to the premade functions like defining a block in which the variable get set to a certain value...

dkxls October 28, 2013 10:31

you can force the writing of a field at any time with
Code:

fracSol.write();
You just should take care that you don't do it in the time-loop, as the field would then be written every time-step. ;)

ThomasV October 28, 2013 11:07

Thanks - if I can come up with enough self-motivation after having finished my master thesis I might write a beginners tutorial so people won't have such a hard time figuring out such "mundane" things like this... ;)

Kummi December 21, 2018 11:39

Hello Foamers,
My problem is almost similar. I have two volScalarFields T (temperature) and k (thermal conductivity). volScalarField k depends upon volScalarField T with the following expression:
Quote:

forAll( TCells, celli )
{
k[celli] = 1.833e-03* Foam::sqrt(TCells[celli]);
}

I need to update the value (k) thermal conductivity for every time step based on Temperature (T). I have followed the above comments and compiled the new FOAM successfully. However, k returns constant value for every time step after solving the expression.
For verification, I am hereby mentioning the source (.C) and createFields.H files.
Quote:

SOURCE FILE

while (simple.loop())
{
Info<< "Time = " << runTime.timeName() << nl << endl;

while (simple.correctNonOrthogonal())
{
{
solve
(
fvm::ddt(rho * Cp, T)
- fvm::laplacian(k, T)
);
scalarField& TCells = T.primitiveFieldRef();
forAll( TCells, celli )
{
k[celli] = 1.833e-03* Foam::sqrt(TCells[celli]);
}
}
}
runTime.write();

Info<< "ExecutionTime = " << runTime.elapsedCpuTime() << " s"
<< " ClockTime = " << runTime.elapsedClockTime() << " s"
<< nl << endl;
}
Quote:

createField FILE
Info<< "Reading field T\n" << endl;

volScalarField T
(
IOobject
(
"T",
runTime.timeName(),
mesh,
IOobject::MUST_READ,
IOobject::AUTO_WRITE
),
mesh
);

Info<< "Reading field k\n" << endl;

volScalarField k
(
IOobject
(
"k",
runTime.timeName(),
mesh,
IOobject::MUST_READ,
IOobject::AUTO_WRITE
),
mesh,
dimensionSet(1,1,-3,-1,0,0,0)
);


Info<< "Reading transportProperties\n" << endl;

IOdictionary transportProperties
(
IOobject
(
"transportProperties",
runTime.constant(),
mesh,
IOobject::MUST_READ_IF_MODIFIED,
IOobject::AUTO_WRITE
)
);

Info<< "Reading density rho\n" << endl;

dimensionedScalar rho
(
transportProperties.lookup("rho")
);

Info<< "Reading thermal capacity Cp\n" << endl;

dimensionedScalar Cp
(
transportProperties.lookup("Cp")
);
Please give me a clue on how to update value of k depending upon T. Kindly share your ideas please. Your thoughts are highly helpful !!!
Thank you

SHUBHAM9595 March 10, 2020 12:51

Quote:

Please give me a clue on how to update value of k depending upon T.
I know it's too late but might help someone else. You just have to assign k in the last section after solving TEqn so that for each time step you can have updated k based on the most recent T.


Code:

  fvScalarMatrix TEqn
    (
        fvm::ddt(T)
      +        fvm::div(phi, T)
      - fvm::laplacian(alpha, T)
    ==
        (1/cp)*(tau && gradU)
    );

TEqn.solve();

k = 1.833e-03* Foam::sqrt(T);

Just make sure you have
Code:

IOobject::AUTO_WRITE
for "k" while declaring it in createFields.H


All times are GMT -4. The time now is 00:07.