forked from diffpy/diffpy.srfit
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathfithook.py
More file actions
257 lines (204 loc) · 6.77 KB
/
fithook.py
File metadata and controls
257 lines (204 loc) · 6.77 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
#!/usr/bin/env python
##############################################################################
#
# diffpy.srfit by DANSE Diffraction group
# Simon J. L. Billinge
# (c) 2008 The Trustees of Columbia University
# in the City of New York. All rights reserved.
#
# File coded by: Chris Farrow
#
# See AUTHORS.txt for a list of people who contributed.
# See LICENSE_DANSE.txt for license information.
#
##############################################################################
"""The FitHook class for inspecting the progress of a FitRecipe
refinement.
FitHooks are called by a FitRecipe during various times of the residual
is evaluation. The default FitHook simply counts the number of times the
residual is called, and reports that number every time the residual is
calculated. Depending on the verbosity, it will also report the residual
and the current variable values.
Custom FitHooks can be added to a FitRecipe with the
FitRecipe.setFitHook method.
"""
from __future__ import print_function
__all__ = ["FitHook"]
import numpy
from diffpy.srfit.util import sortKeyForNumericString
class FitHook(object):
"""Base class for inspecting the progress of a FitRecipe refinement.
Can serve as a fithook for the FitRecipe class (see
FitRecipe.push_fit_hook method.) The methods in this class are
called during the preparation of the FitRecipe for refinement, and
during the residual call. See the class methods for a description of
their purpose.
"""
def reset(self, recipe):
"""Reset the hook data.
This is called whenever FitRecipe._prepare is called, which is
whenever a configurational change to the fit hierarchy takes
place, such as adding a new ParameterSet, constraint or
restraint.
"""
return
def precall(self, recipe):
"""This is called within FitRecipe.residual, before the
calculation.
Parameters
----------
recipe
The FitRecipe instance
"""
return
def postcall(self, recipe, chiv):
"""This is called within FitRecipe.residual, after the
calculation.
Parameters
----------
recipe
The FitRecipe instance
chiv
The residual vector
"""
return
# End class FitHook
class PrintFitHook(FitHook):
"""Base class for inspecting the progress of a FitRecipe refinement.
This FitHook prints out a running count of the number of times the residual
has been called, or other information, based on the verbosity.
Attributes
----------
count
The number of times the residual has been called (default 0).
verbose
An integer telling how verbose to be (default 1).
0
print nothing
1
print the count during the precall
2
print the residual during the postcall
>=3
print the variables during the postcall
"""
def __init__(self):
"""Initialize the attributes."""
self.count = 0
self.verbose = 1
return
def reset(self, recipe):
"""Reset the hook data.
This is called whenever FitRecipe._prepare is called, which is
whenever a configurational change to the fit hierarchy takes
place, such as adding a new ParameterSet, constraint or
restraint.
"""
self.count = 0
return
def precall(self, recipe):
"""This is called within FitRecipe.residual, before the
calculation.
Parameters
----------
recipe
The FitRecipe instance
"""
self.count += 1
if self.verbose > 0:
print(self.count)
return
def postcall(self, recipe, chiv):
"""This is called within FitRecipe.residual, after the
calculation.
Parameters
----------
recipe
The FitRecipe instance
chiv
The residual vector
"""
if self.verbose < 2:
return
# Get the number of restraints
numres = len(recipe._restraintlist)
chi2tot = numpy.dot(chiv, chiv)
chi2 = 0
res = 0
if numres > 0:
chi = chiv[:-numres]
resv = chiv[-numres:]
chi2 = numpy.dot(chi, chi)
res = numpy.dot(resv, resv)
print("Residual:", chi2tot)
if numres:
print("FitContributions:", chi2)
print("Restraints:", res)
if self.verbose >= 3:
print("Variables")
vnames = recipe.get_names()
vals = recipe.get_values()
# byname = _byname()
items = sorted(zip(vnames, vals), key=_byname)
for name, val in items:
print(" %s = %f" % (name, val))
return
def _byname(nv):
return sortKeyForNumericString(nv[0])
# End class PrintFitHook
# TODO - Display the chi^2 on the plot during refinement.
class PlotFitHook(FitHook):
"""This FitHook has live plotting of whatever is being refined."""
def reset(self, recipe):
"""Set up the plot."""
FitHook.reset(self, recipe)
self._plots = []
import pylab
pylab.clf()
pylab.ion()
nc = len(recipe._contributions)
if nc > 1:
ncols = 2
nrows = (nc + 1) / 2
for idx, c in enumerate(recipe._contributions.values()):
name = c.name
xname = c._xname
yname = c._yname
p = c.profile
# Create a subplot
if nc > 1:
pylab.subplot(nrows, ncols, idx + 1)
pdata = pylab.plot(p.x, p.y, "bo")[0]
pfit = pylab.plot(p.x, p.y, "r-")[0]
self._plots.append((pdata, pfit))
pylab.xlabel(xname)
pylab.ylabel(yname)
pylab.title(name)
# Set up some event handling, so things behave nicely.
# def redraw(event):
# canvas = event.canvas
# canvas.draw()
# return
# pylab.connect('resize_event', redraw)
return
def postcall(self, recipe, chiv):
"""This is called within FitRecipe.residual, after the
calculation.
Find data and plot it.
Parameters
----------
recipe
The FitRecipe instance
chiv
The residual vector
"""
FitHook.postcall(self, recipe, chiv)
import pylab
for c, plottup in zip(recipe._contributions.values(), self._plots):
p = c.profile
pdata = plottup[0]
pfit = plottup[1]
pdata.set_data(p.x, p.y)
pfit.set_data(p.x, p.ycalc)
pylab.draw()
return