Diffusion Coefficient¶
The diffusion coefficient (or diffusivity) of a set of \(N\) particles in \(d\) dimensions is given by:
where \(\hat{v}_{n,i}(t)\) is the \(i\)-th Cartesian component of the time-dependent velocity of particle \(n\). For molecular systems, the center-of-mass velocities are typically used.
For a simple fluid, the result is called the self-diffusion coefficient or self-diffusivity. The same expression applies to the diffusion coefficient of components of a mixture or guest molecules in porous media.
Note that this definition is valid only if the particles of interest exhibit diffusive motion. If they oscillate around a fixed center, the zero-frequency component of the velocity autocorrelation spectrum will approach zero, resulting in a diffusion coefficient of zero. This scenario may occur when the diffusion is governed by an activated hopping process, and the simulation is too short to capture such rare events.
The derivation of this result can be found in several references, e.g., Section 4.4.1 of “Understanding Molecular Simulation” by Frenkel and Smit [FS02], Section 7.7 of “Theory of Simple Liquids” by Hansen and McDonald [HM13], or Section 13.3.2 of “Statistical Mechanics: Theory and Molecular Simulation” by Tuckerman [Tuc23].
How to Compute with STACIE?¶
It is assumed that you can load the particle velocities into a 2D NumPy array velocities
.
Each row of this array corresponds to a single Cartesian component of a particle’s velocity, while
each column corresponds to a specific time step.
You should also store the time step in a Python variable.
The diffusion coefficient can then be computed as follows:
import numpy as np
from stacie import compute_spectrum, estimate_acint, plot_results, ExpPolyModel, UnitConfig
# Load all the required inputs, the details of which will depend on your use case.
velocities = ...
timestep = ...
# Computation with STACIE.
# Note that the factor 1/(N*d) is implied:
# the average spectrum over all velocity components is computed.
# Note that the zero-frequency component is usually not reliable
# because typically the total momentum is constrained or conserved.
spectrum = compute_spectrum(
velocities,
prefactors=1.0,
timestep=timestep,
include_zero_freq=False,
)
# The unit configuration assumes SI units are used systematically.
# You may need to adapt this to the units of your data.
uc = UnitConfig(
acint_symbol="D",
acint_unit_str="m$^2$/s",
time_unit=1e-12,
time_unit_str="ps",
freq_unit=1e12,
freq_unit_str="THz",
)
# Actual analysis with STACIE.
result = estimate_acint(spectrum, ExpPolyModel([0, 1, 2]), unit_config=uc, verbose=True)
print("Diffusion coefficient", result.acint)
print("Uncertainty of the diffusion coefficient", result.acint_std)
plot_results("diffusion_coefficient.pdf", result, uc)
A worked example can be found in the notebook Diffusion on a Surface with Newtonian Dynamics.
One can also use particle positions and derive block-averaged velocities by applying something that looks like a finite difference approximation:
where the index \(i\) runs over the recorded time steps and \(\Delta_t\) is time between two recorded steps. Formally, this resembles a finite difference approximation of the velocity, except that \(\Delta_t\) can be large.
Warning
If you have sampled particle velocities directly with a large sampling interval, they cannot be used because they are not proper block averages. In this case, your data violates the Niquist-Shannon sampling theorem, and the results will be perturbed by aliasing artifacts. The block averages satisfy the sampling theorem by construction because they average out the high-frequency components.
For example, if the trajectory data contains positions in Å, recorded every 10 ps, and you want to compute the diffusion coefficient in m²/s, the code would be:
import numpy as np
from stacie import compute_spectrum, estimate_acint, ExpPolyModel, UnitConfig, plot_results
# Define units of external data in "internal" SI base units.
PICOSECOND = 1e-12
ANGSTROM = 1e-10
TERAHERTZ = 1e12
# Load all the required inputs, the details of which will depend on your use case.
# It is assumed that each row of `positions` corresponds to a single Cartesian component
# of a particle's position, while each column corresponds to a specific time step.
positions = (...) * ANGSTROM
timestep = 10.0 * PICOSECOND
# Compute velocities, as if you are using finite difference approximation.
# In fact, these are block-averaged velocities.
velocities = np.diff(positions, axis=1) / timestep
# Computation with STACIE, as before.
spectrum = compute_spectrum(
velocities,
prefactors=1.0,
timestep=timestep,
include_zero_freq=False,
)
uc = UnitConfig(
acint_symbol="D",
acint_unit_str="m$^2$/s",
time_unit=PICOSECOND,
time_unit_str="ps",
freq_unit=TERAHERTZ,
freq_unit_str="THz",
)
result = estimate_acint(spectrum, ExpPolyModel([0, 1, 2]), verbose=True, unit_config=uc)
plot_results("diffusion_coefficient.pdf", result, uc)