#!/usr/bin/env python3"""
Minimal working fit examples: finds radius of sphere in Born approximation.
"""importbornagainasbafrombornagainimportdeg,nm,nm2defget_simulation(P):"""
Returns GISAS simulation for given set of parameters.
"""radius=P["radius"]particle=ba.Particle(ba.RefractiveMaterial("Particle",6e-4,2e-8),ba.Sphere(radius),ba.AlignAt_Bottom)layer_top=ba.Layer(ba.Vacuum())layer_bottom=ba.Layer(ba.Vacuum())layer_top.deposit2D(ba.Dilute2D(0.01/nm2,particle))sample=ba.Sample()sample.addLayer(layer_top)sample.addLayer(layer_bottom)n=100beam=ba.Beam(1,0.1*nm,0.2*deg)detector=ba.SphericalDetector(n,-1*deg,1*deg,n,0.,2*deg)simulation=ba.ScatteringSimulation(beam,sample,detector)returnsimulationdeffake_data():"""
Generating "experimental" data by running simulation with default parameters.
"""simulation=get_simulation({'radius':5*nm})result=simulation.simulate()returnresultif__name__=='__main__':data=fake_data()fit_objective=ba.FitObjective()fit_objective.addFitPair(get_simulation,data)fit_objective.initPrint(10)P=ba.Parameters()P.add("radius",4.*nm,min=0.01)minimizer=ba.Minimizer()result=minimizer.minimize(fit_objective.evaluate,P)fit_objective.finalize(result)
auto/Examples/fit/scatter2d/fit2d.py
Explanation
The function get_sample is basically the same as
in our basic GISAS simulation example,
except that radius and height of the cylindrical disks are now
supplied as external parameters.
These parameters are passed through the function argument params
which can be either a Python dict,
or an instance of the BornAgain class Parameters.
The function get_simulation has the same function argument params,
from which it takes beam intensity and background level.
These two parameters are on a logarithmic scale.
The function start_parameters_1 is supplied along with the model
in order to facilitate the benchmarking of different fith methods.
The FitObjective created here is used to put into correspondence the real data,
represented by a numpy array, and the simulation, represented by the get_simulation callable.
On every fit iteration FitObjective
will generate a new simulation object
for a given set of fit parameter values using the get_simulation callable,
run it to obtain simulated detector intensities,
calculate $\chi^2$ between simulated and real data.
The method fit_objective.evaluate provided by the FitObjective class interface is used here as an objective function.
The method accepts fit parameters and returns the $\chi^2$ value,
calculated between experimental and simulated images for the given values of the fit parameters.
The method is passed to the minimizer together with the initial fit parameter values.
The minimizer.minimize starts a fit that will
continue further without user intervention until the minimum
is found or the minimizer failed to converge.
The rest of the code demonstrates how to access the fit results.