Skip to main content

plotting - Working precision problem in Complex integrand


I'm trying to plot 2d numeric integral of a complex function which is actually real. First problem - small non zero complex part, I was forced to plot RE of it. Second - I'm using only exact numbers but got General::precw warning and can't tune WorkingPrecision! The output interpolation is very rough and I have no clue how to deal with it. My code:


c[x_, y_] := (x*u + (1 - u)*y + 4*u*(1 - u)*k^2)^(1/2)
a = 1
W[r_, k_] := (2 Exp[-2*I*k*r])/(Pi^3 a^3)*

NIntegrate[
Exp[4*I*u*r*k]*
u*(1 - u)*(3/4 Exp[-2*r*c[1, 1]]/c[1, 1]^5 +
3/2 (Exp[-2*r*c[1, 1]]*r)/c[1, 1]^4 + (Exp[-2*r*c[1, 1]]*r^2)/
c[1, 1]^3), {u, 0, 1}, WorkingPrecision -> 20]
Plot3D[r^2*k^2*W[r, k], {r, 0, 5}, {k, 0, 5}, PlotRange -> All,
AxesLabel -> Automatic]

Answer



Some recommendations:




  • Since the integral is provably real, use Re on the integral instead of Chop, since Chop will chop small real parts, too.

  • To speed things up slightly, use Re on the integrand, since NIntegrate will compute with real numbers instead complex ones. Simplify it beforehand so that no complex results are generated in computing the integral.

  • To speed things up further, use a higher setting of Gauss points in the "GaussKronrodRule". The integrand is fairly smooth and not very oscillatory, and higher setting leads to a quickly converging integral.

  • Since it is a well-behaved integral, a high WorkingPrecision is unnecessary. Using MachinePrecision will speed things up even more.

  • The plot is ragged because of scaling: The variation in height ($\approx 10^{-3}$) is so small that the plot is virtually flat (in the standard unscaled Euclidean geometry) and Plot3D does no recursive subdivision, no matter what the setting of MaxRecursion.

  • To fixed the raggedness, one can increase the PlotPoints, which is somewhat wasteful, or scale the function, so that the automatic adaptive algorithm works effectively; then rescale the result.

  • For another speed improvement when plotting, lower the PrecisionGoal in NIntegrate. Since the integrand converges quickly, you might risk being bold and assume the actual error is much less than the error estimate NIntegrate calculates. In any case, three or four digits of precision is usually enough for plotting.

  • There's an error in the coding of W[] and c[]. In W[r, k], the value of k will not be passed to the body of c[1, 1,]. It has the appearance of working because Plot3D[] uses Block to temporarily set the global value of k. Thus, while Plot3D is being computed, the parameter k in W[r, k] and the symbol k in c[1, 1] happen to have the same value. Try evaluating W[2, 3] with the OP's code and you get an error.


OP's refactored code (thanks to @xzczd for suggesting "SymbolicProcessing" -> 0):



ClearAll[c, W];
c[x_, y_, k_] := (x*u + (1 - u)*y + 4*u*(1 - u)*k^2)^(1/2); (* N.B. argument k *)
a = 1;
$pg = Automatic; (* symbol $pg for PrecisionGoal setting in NIntegrate[] *)
Block[{u, k, r}, (* protect symbols while definition is constructed *)
With[{integrand =
Simplify[(2 Exp[-2*I*k*r])/(Pi^3 a^3) Exp[4*I*u*r*k]*
u*(1 - u)*(3/4 Exp[-2*r*c[1, 1, k]]/c[1, 1, k]^5 + (* N.B. argument k *)
3/2 (Exp[-2*r*c[1, 1, k]]*r)/
c[1, 1, k]^4 + (Exp[-2*r*c[1, 1, k]]*r^2)/c[1, 1, k]^3) //

Re // ComplexExpand,
0 < k < 5 && 0 < r < 5 && 0 < u < 1]},
W[r_?NumericQ, k_?NumericQ] := NIntegrate[
integrand,
{u, 0, 1},
Method -> {"GaussKronrodRule", "SymbolicProcessing" -> 0, "Points" -> 11},
PrecisionGoal -> $pg, WorkingPrecision -> MachinePrecision]
]
]


Timing comparison for different precision goals:


Block[{$pg = 3},   (* low setting for  PrecisionGoal  in  NIntegrate[] *)
Table[W[r, k], {r, 0., 5., 0.1}, {k, 0., 5., 0.1}]]; // AbsoluteTiming
(* {4.4409, Null} *)

Block[{$pg = 8}, (* ~default setting for PrecisionGoal in NIntegrate[] *)
Table[W[r, k], {r, 0., 5., 0.1}, {k, 0., 5., 0.1}]]; // AbsoluteTiming
(* {10.9349, Null} *)

The unscaled plot. Raising MaxRecursion has no effect:



 Block[{$pg = 3},
Plot3D[r^2*k^2*W[r, k] // Re, {r, 0, 5},
{k, 0, 5},
MaxRecursion -> 15,
PlotRange -> All, AxesLabel -> Automatic]
]


The scaled plot. The default MaxRecursion is nearly perfect when the plot is scaled so that scale of each axis are approximately the same. Update: I tried ScalingFunctions earlier, but I must have done something wrong. It does work, but in this case, the ticks are not as nice as the manual scaling (see below).


Block[{$pg = 3},

Plot3D[r^2*k^2*W[r, k] // Re, {r, 0, 5},
{k, 0, 5},
ScalingFunctions -> {None, None, {10^4 # &, 10^-4 # &}},
PlotRange -> All, AxesLabel -> Automatic]
]


The previous scaled plot code produces nicer ticks:


 Block[{$pg = 3, scale = 1.*^4},
Plot3D[scale*r^2*k^2*W[r, k] // Re, {r, 0, 5},

{k, 0, 5},
PlotRange -> All, AxesLabel -> Automatic] /.
gc_GraphicsComplex :> Scale[gc, {1., 1., 1/scale}, {0., 0., 0.}]
]


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[ ...