-
Notifications
You must be signed in to change notification settings - Fork 105
/
Copy pathfitlin.h
executable file
·69 lines (65 loc) · 1.76 KB
/
fitlin.h
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
struct Fitlin {
Int ndat, ma;
VecDoub_I &x,&y,&sig;
VecDoub (*funcs)(const Doub);
VecBool ia;
VecDoub a;
MatDoub covar;
Doub chisq;
Fitlin(VecDoub_I &xx, VecDoub_I &yy, VecDoub_I &ssig, VecDoub funks(const Doub))
: ndat(xx.size()), x(xx), y(yy), sig(ssig), funcs(funks) {
ma = funcs(x[0]).size();
a.resize(ma);
covar.resize(ma,ma);
ia.resize(ma);
for (Int i=0;i<ma;i++) ia[i] = true;
}
void hold(const Int i, const Doub val) {ia[i]=false; a[i]=val;}
void free(const Int i) {ia[i]=true;}
void fit() {
Int i,j,k,l,m,mfit=0;
Doub ym,wt,sum,sig2i;
VecDoub afunc(ma);
for (j=0;j<ma;j++) if (ia[j]) mfit++;
if (mfit == 0) throw("lfit: no parameters to be fitted");
MatDoub temp(mfit,mfit,0.),beta(mfit,1,0.);
for (i=0;i<ndat;i++) {
afunc = funcs(x[i]);
ym=y[i];
if (mfit < ma) {
for (j=0;j<ma;j++)
if (!ia[j]) ym -= a[j]*afunc[j];
}
sig2i=1.0/SQR(sig[i]);
for (j=0,l=0;l<ma;l++) {
if (ia[l]) {
wt=afunc[l]*sig2i;
for (k=0,m=0;m<=l;m++)
if (ia[m]) temp[j][k++] += wt*afunc[m];
beta[j++][0] += ym*wt;
}
}
}
for (j=1;j<mfit;j++) for (k=0;k<j;k++) temp[k][j]=temp[j][k];
gaussj(temp,beta);
for (j=0,l=0;l<ma;l++) if (ia[l]) a[l]=beta[j++][0];
chisq=0.0;
for (i=0;i<ndat;i++) {
afunc = funcs(x[i]);
sum=0.0;
for (j=0;j<ma;j++) sum += a[j]*afunc[j];
chisq += SQR((y[i]-sum)/sig[i]);
}
for (j=0;j<mfit;j++) for (k=0;k<mfit;k++) covar[j][k]=temp[j][k];
for (i=mfit;i<ma;i++)
for (j=0;j<i+1;j++) covar[i][j]=covar[j][i]=0.0;
k=mfit-1;
for (j=ma-1;j>=0;j--) {
if (ia[j]) {
for (i=0;i<ma;i++) SWAP(covar[i][k],covar[i][j]);
for (i=0;i<ma;i++) SWAP(covar[k][i],covar[j][i]);
k--;
}
}
}
};