Skip to main content

Environmental

The EnvironmentalExtensions class bridges ScalarField (∇, ∇²) and VectorField to atmospheric and aquatic transport physics — Gaussian plume dispersion, Fickian diffusion, advection, and the advection–diffusion equation.

Namespace: CSharpNumerics.Physics.Environmental


🌫️ Gaussian Plume

Steady-state atmospheric dispersion from a point source with ground reflection:

using CSharpNumerics.Physics.Enums;

double Q = 10.0; // emission rate (kg/s)
var source = new Vector(0, 0, 50); // 50 m stack
var wind = new Vector(1, 0, 0); // wind along +X

// Using Pasquill–Gifford stability class
ScalarField C = Q.GaussianPlume(
windSpeed: 5.0,
stackHeight: 50,
sourcePosition: source,
windDirection: wind,
stability: StabilityClass.D); // neutral conditions

double conc = C.Evaluate(new Vector(500, 0, 0)); // concentration 500 m downwind
Vector grad = C.Gradient((500, 0, 0)); // concentration gradient
double lap = C.Laplacian((500, 0, 0)); // ∇²C

// Quick ground-level centerline concentration
double Cgl = Q.GaussianPlumeGroundLevel(
windSpeed: 5.0, stackHeight: 50,
downwindDistance: 1000,
stability: StabilityClass.C);

Custom dispersion — supply your own σy(x), σz(x):

ScalarField C = Q.GaussianPlume(
windSpeed: 5.0,
stackHeight: 50,
sourcePosition: source,
windDirection: wind,
sigmaY: x => 0.22 * x / Math.Sqrt(1 + 0.0001 * x),
sigmaZ: x => 0.20 * x);

Briggs dispersion parameters are also available directly:

double sy = EnvironmentalExtensions.BriggsSigmaY(1000, StabilityClass.D);
double sz = EnvironmentalExtensions.BriggsSigmaZ(1000, StabilityClass.D);

💨 Transient Gaussian Puff

Time-dependent dispersion of an instantaneous release that advects downwind — the puff centre moves at wind speed while the cloud expands according to Briggs σ(u·t):

double Q = 5.0;  // emission rate (kg/s)
double releaseSeconds = 10.0; // release duration → mass = Q·Δt

ScalarField C = Q.GaussianPuff(
releaseSeconds: releaseSeconds,
windSpeed: 10.0,
stackHeight: 50,
sourcePosition: new Vector(0, 0, 50),
windDirection: new Vector(1, 0, 0),
time: 30.0, // seconds since release
stability: StabilityClass.D);

// At t=30s, puff centre is at x = u·t = 300 m downwind
double conc = C.Evaluate(new Vector(300, 0, 50)); // peak region

// Evaluate at different times for animation
for (double t = 10; t <= 120; t += 10)
{
var puff = Q.GaussianPuff(releaseSeconds, 10, 50,
new Vector(0, 0, 50), new Vector(1, 0, 0), t, StabilityClass.D);
double peak = puff.Evaluate(new Vector(10 * t, 0, 50));
}

🧪 Diffusion (Fick's Laws)

var C = new ScalarField(r => Math.Exp(-(r.x * r.x + r.y * r.y)));
double D = 0.01; // m²/s

// Fick's first law: diffusive flux J = −D∇C → VectorField
VectorField J = C.DiffusionFlux(D);

// Fick's second law: ∂C/∂t = D∇²C → ScalarField
ScalarField dCdt = C.DiffusionRate(D);
double rate = dCdt.Evaluate((0.5, 0, 0));

Analytical point-source diffusion — 3D Gaussian spreading:

// Mass M released at origin, evaluated at time t
ScalarField C = 1.0.DiffusionPointSource(
diffusionCoefficient: 0.01,
time: 100,
sourcePosition: new Vector(0, 0, 0));
// C(r,t) = M / (4πDt)^(3/2) · exp(−|r|² / (4Dt))

💨 Advection

var C = new ScalarField(r => Math.Exp(-r.x * r.x));
var wind = new VectorField(r => 2.0, r => 0, r => 0); // uniform 2 m/s along X

// ∂C/∂t = −v·∇C
ScalarField dCdt = C.AdvectionRate(wind);

🌊 Advection–Diffusion

Combined transport with optional source term:

var C = new ScalarField(r => Math.Exp(-(r.x * r.x + r.y * r.y)));
var wind = new VectorField(r => 1.0, r => 0.5, r => 0);
double D = 0.1;

// ∂C/∂t = D∇²C − v·∇C
ScalarField dCdt = C.AdvectionDiffusionRate(wind, D);

// With source term S(r)
var source = new ScalarField(r => r.x > 0 && r.x < 1 ? 0.5 : 0);
ScalarField dCdt2 = C.AdvectionDiffusionRate(wind, D, source);

Péclet number — determines transport regime:

double Pe = 2.0.PecletNumber(characteristicLength: 100, diffusionCoefficient: 0.1);
// Pe = 2000 → advection-dominated

🔥 Rothermel Surface Fire Spread

The RothermelModel static class implements the Rothermel (1972) surface fire spread model — the standard used by FARSITE, FlamMap, and BehavePlus. All parameters are in SI units.

Namespace: CSharpNumerics.Physics.Environmental.Fire

Rate of spread:

R=IRξ(1+ϕw+ϕs)ρbεQigR = \frac{I_R \cdot \xi \cdot (1 + \phi_w + \phi_s)}{\rho_b \cdot \varepsilon \cdot Q_{ig}}

using CSharpNumerics.Physics.Environmental.Fire;
using CSharpNumerics.Physics.Materials.Fire;
using CSharpNumerics.Physics.Materials.Fire.Enums;

// Get a standard Anderson 13 fuel model
FuelModel grass = FuelLibrary.Get(FuelModelType.ShortGrass);

// Rate of spread on flat terrain, moderate wind
double ros = RothermelModel.RateOfSpread(
fuel: grass,
moistureContent: 0.05, // 5% dead fuel moisture
windSpeed: 2.24, // m/s (≈ 5 mph mid-flame)
slopeRadians: 0); // flat
// ros ≈ 23–25 m/min for Short Grass

// Individual sub-equations
double IR = RothermelModel.ReactionIntensity(grass, moistureContent: 0.05);
double phiW = RothermelModel.WindFactor(grass, midflameWindSpeed: 2.24);
double phiS = RothermelModel.SlopeFactor(
RothermelModel.PackingRatio(grass), slopeRadians: Math.PI / 6);
double xi = RothermelModel.PropagatingFluxRatio(grass);
double Qig = RothermelModel.HeatOfPreignition(moistureContent: 0.05);
double eps = RothermelModel.EffectiveHeatingNumber(grass);

// Flame length from Byram's fireline intensity correlation
double fl = RothermelModel.FlameLength(IR, ros); // metres

Sub-equation reference:

MethodSymbolUnit
ReactionIntensityIRI_RkJ/m²·min
WindFactorϕw\phi_w
SlopeFactorϕs\phi_s
PropagatingFluxRatioξ\xi
HeatOfPreignitionQigQ_{ig}kJ/kg
EffectiveHeatingNumberε\varepsilon
PackingRatioβ\beta
OptimalPackingRatioβop\beta_{op}
FlameLengthLLm

💧 Water Hydraulics & Contaminant Data

The Physics.Environmental.Water namespace provides open-channel hydraulics (Manning's equation) and longitudinal dispersion for river transport modelling. The Physics.Materials.Water namespace defines aquatic contaminant descriptors with decay, adsorption, and toxicity properties.

Manning's Equation — Open-Channel Velocity

u=1nRh2/3S01/2u = \frac{1}{n} R_h^{2/3} S_0^{1/2}

using CSharpNumerics.Physics.Environmental.Water;

// Rectangular channel: W=10 m, H=2 m, Manning's n=0.035, slope=0.001
double Rh = ManningEquation.RectangularHydraulicRadius(10, 2);
double u = ManningEquation.Velocity(0.035, Rh, 0.001); // ≈ 1.14 m/s
double Q = ManningEquation.Discharge(0.035, Rh, 0.001, 20); // 20 m² area

// Trapezoidal channel
double RhTrap = ManningEquation.TrapezoidalHydraulicRadius(8, 2, 1.5);
double uTrap = ManningEquation.Velocity(0.03, RhTrap, 0.0005);

Fischer Longitudinal Dispersion

EL=0.011u2W2HuE_L = 0.011 \frac{u^2 W^2}{H \cdot u_*}

double uStar = LongitudinalDispersion.ShearVelocity(Rh, 0.001);
double EL = LongitudinalDispersion.FischerCoefficient(u, 10, 2, uStar);

// Decay & retardation helpers
double lambda = LongitudinalDispersion.DecayConstant(halfLifeSeconds: 9.5e8); // Cs-137
double Rf = LongitudinalDispersion.RetardationFactor(
bulkDensity: 1600, Kd: 50, porosity: 0.4); // Rf > 1 → retarded transport

Tributary Mixing

// Two rivers merge: main at 80 mg/L (Q=20 m³/s) + tributary at 0 mg/L (Q=10 m³/s)
double downstreamConc = MixingZoneModel.TributaryMixing(80, 20, 0, 10);
// → 53.3 mg/L (mass-balance dilution)

double mixLength = MixingZoneModel.MixingLength(1.0, 10, 0.6); // ≈ 116.7 m