Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Species Sensitivity Analysis for 1D Flames #1830

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

marina8888
Copy link

@marina8888 marina8888 commented Dec 21, 2024

This PR is for the addition of a 1D species sensitivity analysis in the Python interface. Closes the long overdue Enhancement 71.

I have another version considering distance as input, but this requires some additional logic and for error handling for distances outside of the domain.

Tested code on ImpingingJet flame and FreeFlame to confirm the behaviour is as expected. Please see an example code for usage.

  import cantera as ct
  import matplotlib.pyplot as plt
  import matplotlib
  matplotlib.use('macosx')
  import pandas as pd
  # Define the gas mixture
  gas = ct.Solution("gri30.yaml")  # Use the GRI-Mech 3.0 reaction mechanism
  gas.TP = 295, ct.one_atm  # Methane-air mixture
  air = "O2:0.21,N2:0.79"
  gas.set_equivalence_ratio(phi=1.0, fuel="NH3:0.7, H2:0.3", oxidizer=air)
  flame = ct.ImpingingJet(gas=gas, width = 0.02)
  flame.inlet.mdot = 0.35 * gas.density
  flame.surface.T = 493.5
  flame.set_initial_guess("equil")

  # Refine grid to improve accuracy
  flame.set_refine_criteria(ratio=3, slope=0.1, curve=0.1)

  # Solve the flame
  flame.solve(loglevel=1, auto=True)  # Increase loglevel for more output

  # plot temperature:
  # Extract temperature and distance
  temperature = flame.T
  distance = flame.grid  # Grid points (distance in meters)

  # Plot temperature profile
  plt.figure(figsize=(8, 6))
  plt.plot(distance * 1e3, temperature, label="Flame Temperature", color="red")
  plt.xlabel("Distance (mm)")
  plt.ylabel("Temperature (K)")
  plt.title("Temperature Profile of a Flame")
  plt.grid(True, linestyle='--', alpha=0.7)
  plt.legend()
  plt.tight_layout()
  plt.show()


  # Create a DataFrame to store sensitivity-analysis data
  sens = pd.DataFrame(index=gas.reaction_equations(), columns=["sensitivity"])

  # Use the adjoint method to calculate sensitivities
  sens.sensitivity = flame.get_species_reaction_sensitivities("N2O", -1)

  sens.head(10)
  sens = sens.iloc[(-sens['sensitivity'].abs()).argsort()]
  fig, ax = plt.subplots()

  sens.head(15).plot.barh(ax=ax, legend=None)
  ax.invert_yaxis()  # put the largest sensitivity on top
  ax.set_title("Sensitivities for N2O Using GRI Mech")
  ax.set_xlabel(r"Sensitivity: $\frac{\partial\:\ln X_{N2O}}{\partial\:\ln k}$")
  ax.grid(axis='x')
  plt.tight_layout()
  plt.show()

@marina8888 marina8888 changed the title Cantera/enhancements/71 - Addition of Species Sensitivity Analysis for 1D Flames Species Sensitivity Analysis for 1D Flames Dec 21, 2024
Copy link
Member

@ischoegl ischoegl left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@marina8888 ... thank you for the PR! Apart from minor formatting comments, we usually request unit tests for our pytest test suite for new additions. Likewise, it would be great if you could add an example that illustrates species sensitivity analysis (something based on what you have posted in your PR description would be perfectly adequate). For some pointers, please refer to Cantera's Contributing Guidelines. Beyond that, we'd be happy to provide assistance for this PR in case you should have any questions!

def g(sim):
return self.X[self.gas.species_index(species), ix]

Nvars = sum(D.n_components * D.n_points for D in self.domains)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
Nvars = sum(D.n_components * D.n_points for D in self.domains)
n_vars = sum(dd.n_components * dd.n_points for dd in self.domains)

Uppercase letters should be avoided (Cantera mostly follows the PEP8 Python convention with the exception of thermodynamic properties, i.e. T, P, X/Y etc.). There are a couple of other instances above and below. Single-letter variable/function names should likewise be avoided.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants