Finding the Tortuosity (\(\tau\)) of an Image Using tortuosity_fd¶
In this tutorial, we will walk through how to use tortuosity_fd to calculate the tortuosity of an image using a finite difference method. The function takes a binary image to analyze with “True” to indicate the phase of interest as well as the axis along which to find the tortuosity.
Algortihm Description¶
tortuoisty_fd
starts by calculating the porosity of the image.
$\( \epsilon_{orginal} = \frac{\sum_{i = 0}^{N_{x}}\sum_{j = 0}^{N_{y}}im_{ij}}{N_{x}\cdot N_{y}} \)$The second step is to remove non-percolating paths between the inlet and the outlet. This is done by using
trim_nonpercolating_paths
. The description of this filter can be found here.The new porosity is calculated after trimming the non-percolating pores using the same equation as step 1.
$\( \epsilon_{eff} = \frac{\sum_{i = 0}^{N_{x}}\sum_{j = 0}^{N_{y}}im_{ij}}{N_{x}\cdot N_{y}} \)$A cubic network is generated using openpnm and is used as an orthogonal grid.
A dummy phase is created, and openpnm’s Fickian diffusion algorithm (
op.algorithms.FickianDiffusion
) is applied.The inlet concentration and throat diffusive conductance are set to 1.0, and the outlet concentration is set to 0.
Using the rate calculated from the Fickian diffusion algorithm, the effective diffusion coefficient is then calculated from the formula:
$\( D_{Eff} = \frac{\dot{\vec{N}} \cdot (L-1)}{A \cdot \Delta C} \)$The subsequent tortuosity is finally calculated using:
$\(\tau = \frac{D_{AB}}{D_{Eff}} \cdot \varepsilon_{Eff} \)$All useful results are then compiled into a Results object.
Importing Packages¶
import matplotlib.pyplot as plt
import porespy as ps
import numpy as np
ps.visualization.set_mpl_style()
[20:07:45] ERROR PARDISO solver not installed, run `pip install pypardiso`. Otherwise, _workspace.py:56 simulations will be slow. Apple M chips not supported.
Generating the image:¶
For the purposes of this tutorial, we will generate a 200 x 200 pixel image with a target porosity of 0.65.
Running the algorithm¶
As mentioned at the start of the tutorial, the only two inputs for the function are the image and the axis along which to run the calculation. For the x-axis we assign axis a value of 1 and for the y-axis we assign axis a value of 0.
results = ps.simulations.tortuosity_fd(im=im, axis=1)
print(results)
――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――
Results of tortuosity_fd generated at Thu Jan 16 20:07:46 2025
――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――
im Array of size (200, 200)
tortuosity 1.8776173160636371
formation_factor 2.9043927701204795
original_porosity 0.646575
effective_porosity 0.646475
concentration Array of size (200, 200)
time 0.620197809
――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――――
The function outputs an object with several attributes:
Attribute |
Description |
---|---|
|
The calculated tortuosity is given by the equation: |
|
The effective porosity of the image after removing disconnected voxels |
|
The porosity of the image as inputted |
|
The formation factor is given by the equation: |
|
Returns an image containing the concentration values from the simulation |
Calling Values From The Output¶
There are a couple ways to call the values from the returned object. The easiest way to call the values is to use object.attribute