Skip to main content

fitting - NoLinearModelFit error message


to fit my data, I use the following code. Using Manipulate on my fitting function, I can guess roughly the value for my initial fitting parameter which are in the range expected. However, When I try to use NoLinearModelFit, I get wrong fitting parameters and the following message...I don't know how to fix this problem...



NonlinearModelFit: The step size in the search has become less than the tolerance prescribed by the PrecisionGoal option, but the gradient is larger than the tolerance specified by the AccuracyGoal option. There is a possibility that the method has stalled at a point that is not a local minimum.



data = Import["https://pastebin.com/raw/ADajyTYF", "TSV"]

dataT = Transpose[data];

dataT = {10*dataT[[1]], dataT[[2]]};
error = Transpose[data][[3]];
data = Transpose[dataT];
nmax = Length[data];

mu0 := 4*Pi*10^(-7)
Ms := 1.75
lD[DD_] := 10^9*2*DD*mu0/Ms^2

h2[Hp_, q_, RH_] := Hp^2/(1 + q^2*RH^2)^2


Heff[q_, H_, A_] := H + 2*A/(Ms/mu0)*q^2*10^(18)
p[q_, H_, A_] := Ms/Heff[q, H, A]

nenner[q_, H_, DD_, A_] := 1 - p[q, H, A]^2*lD[DD]^2*q^2
chi[Hp_, q_, H_, DD_, RH_, A_] := (4*p[q, H, A]^3*h2[Hp, q, RH]*lD[DD]*q)/nenner[q, H, DD, A]^2

Manipulate[Plot[chi[Hp, q, 5, DD, RH, A], {q, 0, 1.2}, PlotRange -> All, Epilog -> Point[data]], {{Hp, 100}, 0, 2000}, {{DD, 3*10^-3}, 10^-4,10^-1}, {{RH, 40}, 0, 200}, {{A, 2.8*10^-11}, 5*10^-11, 1*10^-11}]



singlefit1 = Normal[NonlinearModelFit[data, chi[Hp, q, 5, DD, RH, A], {{Hp, 100}, {DD, 0.001}, {RH, 40}, {A, 28*10^(-12)}}, q, MaxIterations -> Infinity, Method -> "LevenbergMarquardt",Weights -> error]];
singlefit2 = NonlinearModelFit[data, chi[Hp, q, 5, DD, RH, A], {{Hp, 100}, {DD, 0.001}, {RH, 40}, {A, 28*10^(-12)}}, q, MaxIterations -> Infinity, Method -> "LevenbergMarquardt", Weights -> error];

Print[singlefit2["ParameterTable"]]

Answer



Mathematica is not the problem nor is the problem easily fixable: your model is way over-parameterized (i.e., too complex for the available data).


The symptoms are four-fold: (1) lack of convergence after many iterations, (2) the resulting fit looks OK, (3) The estimates of the parameters are not what you expect (either in sign or magnitude) and (4) the estimated correlation matrix shows several redundant parameters.


Show[ListPlot[data], Plot[singlefit2[q], {q, Min[data[[All, 1]]], Max[data[[All, 1]]]}]]

Data and fit



singlefit2["CorrelationMatrix"] // MatrixForm

$$\left( \begin{array}{cccc} 1. & 1. & -1. & 0.957502 \\ 1. & 1. & -1. & 0.957502 \\ -1. & -1. & 1. & -0.957502 \\ 0.957502 & 0.957502 & -0.957502 & 1. \\ \end{array} \right)$$


If the model form is based on supported theory, then your data can't separate out the individual parameters. If you just need a reasonable prediction based on q, then taking logs and fitting a straight line (i.e., estimating just 3 parameters rather than 5 (not forgetting that the residual error variance is also a parameter to be estimated) works fine (although the error structure could be better modeled):


singlefit3 = NonlinearModelFit[Log[data], a + b q, {{a, -5}, {b, -3}}, q];
singlefit3["BestFitParameters"]
(* {a -> -4.78975, b -> -3.31985} *)
Show[ListLogLogPlot[data],
Plot[Exp[singlefit3[Log[q]]], {q, Min[data[[All, 1]]],
Max[data[[All, 1]]]}, ScalingFunctions -> {"Log", "Log"}]]


Data and fit on log-log scale


And the data and predicted value back on the original scale:


Show[ListPlot[data],
Plot[Exp[singlefit3[Log[q]]], {q, Min[data[[All, 1]]],
Max[data[[All, 1]]]}]]

Data and fit backtransformed to original scale


Comments

Popular posts from this blog

front end - keyboard shortcut to invoke Insert new matrix

I frequently need to type in some matrices, and the menu command Insert > Table/Matrix > New... allows matrices with lines drawn between columns and rows, which is very helpful. I would like to make a keyboard shortcut for it, but cannot find the relevant frontend token command (4209405) for it. Since the FullForm[] and InputForm[] of matrices with lines drawn between rows and columns is the same as those without lines, it's hard to do this via 3rd party system-wide text expanders (e.g. autohotkey or atext on mac). How does one assign a keyboard shortcut for the menu item Insert > Table/Matrix > New... , preferably using only mathematica? Thanks! Answer In the MenuSetup.tr (for linux located in the $InstallationDirectory/SystemFiles/FrontEnd/TextResources/X/ directory), I changed the line MenuItem["&New...", "CreateGridBoxDialog"] to read MenuItem["&New...", "CreateGridBoxDialog", MenuKey["m", Modifiers-...

How to thread a list

I have data in format data = {{a1, a2}, {b1, b2}, {c1, c2}, {d1, d2}} Tableform: I want to thread it to : tdata = {{{a1, b1}, {a2, b2}}, {{a1, c1}, {a2, c2}}, {{a1, d1}, {a2, d2}}} Tableform: And I would like to do better then pseudofunction[n_] := Transpose[{data2[[1]], data2[[n]]}]; SetAttributes[pseudofunction, Listable]; Range[2, 4] // pseudofunction Here is my benchmark data, where data3 is normal sample of real data. data3 = Drop[ExcelWorkBook[[Column1 ;; Column4]], None, 1]; data2 = {a #, b #, c #, d #} & /@ Range[1, 10^5]; data = RandomReal[{0, 1}, {10^6, 4}]; Here is my benchmark code kptnw[list_] := Transpose[{Table[First@#, {Length@# - 1}], Rest@#}, {3, 1, 2}] &@list kptnw2[list_] := Transpose[{ConstantArray[First@#, Length@# - 1], Rest@#}, {3, 1, 2}] &@list OleksandrR[list_] := Flatten[Outer[List, List@First[list], Rest[list], 1], {{2}, {1, 4}}] paradox2[list_] := Partition[Riffle[list[[1]], #], 2] & /@ Drop[list, 1] RM[list_] := FoldList[Transpose[{First@li...

plotting - How to draw lines between specified dots on ListPlot?

I would like to create a plot where I have unconnected dots and some connected. So far, I have figured out how to draw the dots. My code is the following: ListPlot[{{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4,13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full] I have thought using ListLinePlot command, but I don't know how to specify to the command to draw only selected lines between the dots. Do have any suggestions/hints on how to do that? Thank you. Answer One possibility would be to use Epilog with Line : ListPlot[ {{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4, 13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full, Epilog -> { Line[ ...