Skip to content

Latest commit

 

History

History
192 lines (133 loc) · 7.46 KB

diffraction_objects_example.rst

File metadata and controls

192 lines (133 loc) · 7.46 KB
tocdepth:-1

Diffraction Objects Example

This example will demonstrate how to use the functions in the diffpy.utils.diffraction_objects module to create a DiffractionObject instance and analyze your diffraction data using relevant functions.

To create a DiffractionObject, you need to specify the type of the independent variable (referred to as xtype, e.g., one of q, tth, or d), the xarray of the x values, and a yarray of the corresponding intensity values. It is strongly encouraged to specify the wavelength in order to access most of the other functionalities in the class. Additionally, you can specify the type of your scattering experiment using the scat_quantity parameter, the name of your diffraction object using the name parameter, and a metadata dictionary containing relevant information about the data. Here's an example:

import numpy as np
from diffpy.utils.diffraction_objects import DiffractionObject

x = np.array([0.12, 0.24, 0.31, 0.4])  # independent variable (e.g., q)
y = np.array([10, 20, 40, 60])  # intensity values
metadata = {
    "sample": "rock salt from the beach",
    "composition": "NaCl",
    "temperature": "300 K,",
    "experimenters": ["Phil", "Sally"]
}

my_do = DiffractionObject(
     xarray=x,
     yarray=y,
     xtype="q",
     wavelength=1.54,
     scat_quantity="x-ray",
     name="beach_rock_salt_1",
     metadata=metadata
)

By creating a DiffractionObject instance, you store not only the diffraction data but also all the associated information for analysis.

DiffractionObject automatically populates the xarray onto each of q, tth, and d-spacing. Let's say you want to plot your data vs. Q. To do this you would type

import matplotlib.pyplot as plt

plt.plot(my_do.on_q()[0], my_do.on_q()[1])

and to plot the same data vs. two-theta type

plt.plot(my_do.on_tth()[0], my_do.on_tth()[1])

These on_q(), on_tth(), etc., methods return a list with the x-array as the first element and the intensity array as the second element.

We can also accomplish the same thing by passing the xtype as a string to the on_xtype() method, i.e.,

data_on_q = my_do.on_xtype("q")
data_on_tth = my_do.on_xtype("tth")
data_on_d = my_do.on_xtype("d")
plt.plot(data_on_d[0], data_on_d[1])

This makes it very easy to compare a diffraction pattern that was measured or calculated on one xtype with one that was measured or calculated on another. E.g., suppose that you have a calculated powder pattern from a CIF file that was calculated on a d-spacing grid using some software package, and you want to know if a diffraction pattern you have measured on a Q-grid is the same material. You could simply load them both as diffraction objects and plot them together on the same grid.

calculated = DiffractionObject(xcalc, ycalc, "d")
measured = DiffractionObject(xmeas, ymeas, "tth", wavelength=0.717)
plt.plot(calculated.on_q()[0], calculated.on_q()[1])
plt.plot(measured.on_q()[0], measured.on_q()[1])
plt.show()

Now, let's say that these two diffraction patterns were on very different scales. The measured one has a peak intensity of 10,000, but the calculated one only goes to 1. With diffraction objects this is easy to handle. We choose a point on the x-axis where we want to scale the two together and we use the scale_to() method,

Continuing the example above, if we wanted to scale the two patterns together at a position Q=5.5 inverse angstroms, where for the sake of argument we assume the calculated curve has a strong peak, we would replace the code above with

plt.plot(calculated.on_q()[0], calculated.on_q()[1])
plt.plot(measured.scale_to(calculated, q=5.5).on_q()[0], measured.scale_to(calculated, q=5.5).on_q()[1])
plt.show()

The scale_to() method returns a new DiffractionObject which we can assign to a new variable and make use of.

The default behavior is to align the objects based on the maximal value of each diffraction object.

scaled_measured = measured.scale_to(calculated)

If this doesn't give the desirable results, you can specify an xtype=value to scale based on the closest x-value in both objects. For example:

scaled_measured = measured.scale_to(calculated, q=5.5)

For convenience, you can also apply an offset to the scaled new diffraction object with the optional offset argument, for example,

scaled_and_offset_measured = measured.scale_to(calculated, q=5.5, offset=0.5)

DiffractionObject convenience functions

  1. create a copy of a diffraction object using the copy method when you want to preserve the original data while working with a modified version.
copy_of_calculated = calculated.copy()
  1. test the equality of two diffraction objects. For example,
diff_object2 = diff_object1.copy()
diff_object2 == diff_object1

will return True.

  1. make arithmetic operations on the intensities of diffraction objects. e.g.,
doubled_object = 2 * diff_object1     # Double the intensities
sum_object = diff_object1 + diff_object2    # Sum the intensities
subtract_scaled = diff_object1 - 5 * diff_object2   # subtract 5 * obj2 from obj 1
  1. get the value of the DiffractionObject at a given point in one of the xarrays
tth_ninety_index = diff_object1.get_array_index(90, xtype="tth")
intensity_at_ninety = diff_object1.on_tth()[1][tth_ninety_index]

If you do not specify an xtype, it will default to the xtype used when creating the DiffractionObject. For example, if you have created a DiffractionObject called do with xtype="q", you can find its closest index for q=0.25 by typing either of the following:

print(do._input_xtype)     # remind ourselves which array was input.  prints "q" in this case.
index = do.get_array_index(0.25) # no xtype passed, defaults to do._input_xtype, or in this example, q
index = do.get_array_index(0.25, xtype="q") # explicitly choose an xtype to specify a value

5) The dump function saves the diffraction data and relevant information to an xy format file with headers (widely used chi format used, for example, by Fit2D and diffpy. These files can be read by LoadData() in diffpy.utils.parsers).

You can choose which of the data axes (q, tth, or d) to export, with q as the default.

# Assume you have created a Diffraction Object do
file = "diffraction_data.chi"
do.dump(file, xtype="q")

In the saved file diffraction_data.chi, relevant metadata are also written in the header (username, name, scattering quantity, metadata, etc.). The datetime when the DiffractionObject was created and the version of the software (see the Section on get_package_info() for more information) is automatically recorded as well. The diffraction data is saved as two columns: the q values and corresponding intensity values. This ensures your diffraction data, along with all other information, is properly documented and saved for future reference.