use super::Pets;
use crate::hard_sphere::HardSphereProperties;
use feos_core::StateHD;
use num_dual::DualNum;
use std::f64::consts::{FRAC_PI_3, PI};
pub const A: [f64; 7] = [
0.690603404,
1.189317012,
1.265604153,
-24.34554201,
93.67300357,
-157.8773415,
96.93736697,
];
pub const B: [f64; 7] = [
0.664852128,
2.10733079,
-9.597951213,
-17.37871193,
30.17506222,
209.3942909,
-353.2743581,
];
impl Pets {
pub fn dispersion_helmholtz_energy_density<D: DualNum<f64> + Copy>(
&self,
state: &StateHD<D>,
) -> D {
let n = self.sigma.len();
let rho = &state.partial_density;
let d3 = self.hs_diameter(state.temperature).map(|d| d.powi(3));
let eta = rho.component_mul(&d3).sum() * 0.5 * FRAC_PI_3;
let mut rho1mix = D::zero();
let mut rho2mix = D::zero();
for i in 0..n {
for j in 0..n {
let eps_ij = state.temperature.recip() * self.epsilon_k_ij[(i, j)];
let sigma_ij = self.sigma_ij[(i, j)].powi(3);
rho1mix += rho[i] * rho[j] * eps_ij * sigma_ij;
rho2mix += rho[i] * rho[j] * eps_ij * eps_ij * sigma_ij;
}
}
let mut i1 = D::zero();
let mut i2 = D::zero();
let mut eta_i = D::one();
for i in 0..=6 {
i1 += eta_i * A[i];
i2 += eta_i * B[i];
eta_i *= eta;
}
let c1 = ((eta * 8.0 - eta.powi(2) * 2.0) / (eta - 1.0).powi(4) + 1.0).recip();
(-rho1mix * i1 * 2.0 - rho2mix * c1 * i2) * PI
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::pets::parameters::utils::{
argon_krypton_parameters, argon_parameters, krypton_parameters,
};
use approx::assert_relative_eq;
use nalgebra::dvector;
#[test]
fn mix() {
let argon = Pets::new(argon_parameters());
let krypton = Pets::new(krypton_parameters());
let mix = Pets::new(argon_krypton_parameters());
let t = 250.0;
let v = 2.5e28;
let n = 1.0;
let s = StateHD::new(t, v, &dvector![n]);
let a1 = argon.dispersion_helmholtz_energy_density(&s);
let a2 = krypton.dispersion_helmholtz_energy_density(&s);
let s1m = StateHD::new(t, v, &dvector![n, 0.0]);
let a1m = mix.dispersion_helmholtz_energy_density(&s1m);
let s2m = StateHD::new(t, v, &dvector![0.0, n]);
let a2m = mix.dispersion_helmholtz_energy_density(&s2m);
assert_relative_eq!(a1, a1m, epsilon = 1e-14);
assert_relative_eq!(a2, a2m, epsilon = 1e-14);
}
}