Skip to main content

calculus and analysis - Why do I get a different value when I change the order of integration?


I think the following two-dimensional integrals should be equal, since they both integrate the function over the half plane defined by $t>\tau$.


$$\int_{-\infty}^\infty \mathrm{d}t \int_{-\infty}^t \mathrm{d}\tau f(t,\tau) =\int_{-\infty}^\infty \mathrm{d}\tau \int_{\tau}^\infty \mathrm{d}t f(t,\tau) $$


However, when I take the following numerical integral in these two different ways, I get two different answers (only one of which matches my analytic solution).



integrand = Exp[I x1 t-I x2 τ-τ^2/(2 σ^2)] /. {x1 -> 1.0 + .1I, x2 -> 1.2, σ -> 0.1};
NIntegrate[integrand, {t, -∞, ∞}, {Ï„, -∞, t}]
(* 0.201597 + 0.107977 I *)

NIntegrate[integrand, {Ï„, -∞, ∞}, {t, Ï„, ∞}]
(* 0.0247648 + 0.248149 I *)

Any idea why this is the case?


edit: The problem is not only with NIntegrate, but is reproduced by Integrate as well. Running


integrand = Exp[I x1 t-I x2 τ-τ^2/(2 σ^2)];

Integrate[integrand, {Ï„, -∞, ∞}, {t, Ï„, ∞},Assumptions->Re[σ^2]>0&&Im[x1]>0]

gives $\frac{i \sqrt{2 \pi } \sigma e^{-\frac{1}{2} \sigma ^2 (\text{x1}-\text{x2})^2}}{\text{x1}}$ for an answer, which matches what I got when I worked it by hand. But if I do it with the original integration limits


integrand = Exp[I x1 t-I x2 τ-τ^2/(2 σ^2)];
Integrate[integrand, {t, -∞, ∞}, {Ï„, -∞, t},Assumptions->Re[σ^2]>0&&Im[x1]>0]

the evaluation hangs, and eventually the kernel stops working if I don't Abort.


For some reason, the integral can only be evaluated properly both ways if exact numbers are used in defining integrand, as pointed out in Mr.Wizard's answer. I would still like to find out why this is such a tricky integral for Mathematica to do one way but not the other.



Answer



Why the order could matter



Superficially, picking some values for t, integrating Ï„, and finally integrating the results over t is a different calculation than calculating it in a different order. Now there are a few things to explain and investigate to show that this naive observation has a bearing on the OP's integral. Broadly, I would say that the lesson is that two calculations are not always computationally equivalent even if they are mathematically equivalent.


Broadly, integration involves picking sample points, evaluating the integrand at them, and summing the results according to some integration rule. The order of t and Ï„ does not enter into this view and showing that it does in fact matter in the OP's code is something we will show below. Even if it does matter, the question arises why Mathematica does not converge on the correct answer. This seems to be related to two things. One is that the integral over Ï„ as a function of t has quite different properties than the integral over t as a function of Ï„. The other is that the default sampling leads Mathematica to miss this difference; consequently the error estimate calculated by NIntegrate must be faulty, since no warning is issued.


I am not familiar with the underlying mathematics of the multidimensional strategies used by NIntegrate. This answer is based on observations of the behavior of NIntegrate.


Some properties of the integrand


The dominant factor of the integrand is E^(-50 Ï„^2), which means that the greatest contribution to the integral occurs near Ï„ == 0. Less important but still significant is the fact E^(-(t/10)), which vanishes as t increases.


The integral over Ï„ as a function of t is oscillatory, whereas the integral over t as a function of Ï„ is bell-shaped. Oscillatory integrands often require special techniques, but in my mind, I'm not convinced this is as significant as the sampling issues discussed below.


integrand = 
Exp[I x1 t - I x2 τ - τ^2/(2 σ^2)] /.
{x1 -> 1 + 1/10 I, x2 -> 12/10, σ -> 1/10};


ia = Exp[I x1 t] /.
{x1 -> 1 + 1/10 I, x2 -> 12/10, σ -> 1/10};
ib = Exp[-I x2 τ - τ^2/(2 σ^2)] /.
{x1 -> 1 + 1/10 I, x2 -> 12/10, σ -> 1/10};
ia ib == integrand
(* True *)

f[t_] := Integrate[ib, {Ï„, -∞, t}]

g[Ï„_] := Integrate[ia, {t, Ï„, ∞}]


Plot[Evaluate[{Re[#], Im[#]} &[ia f[t]]], {t, -1, 25}]

Mathematica graphics


Plot[Evaluate[{Re[#], Im[#]} &[ib g[Ï„]]], {Ï„, -1/2, 1/2}, 
PlotRange -> All]

Mathematica graphics


Sampling


Sampling points from four calculations at three different plotranges zooming in on the origin. Explanation and code to follow.



graphics


One can see in the sampling points that with the default method, integration is done by sampling along lines where the first variable to NIntegrate is constant. There are other biases in the sampling. In a half-infinite interval, sampling starts near the finite endpoint. In a doubly infinite interval, sampling starts near 0. In both cases the initial sampling points move away from the starting point exponentially. Subsequent refinement depends on the integrand and error estimation. Here is the important point:


The sampling can miss an important region if the region is relatively small and occurs far away from the starting point of the sampling.


For instance, when the τ is integrated first (what I call the dτ integral), sampling is biased toward the finite endpoint at t, and the vertically collinear points reflect the dτ integration. As t varies, the initial sampling points vary along the diagonal line τ == t. This diagonal bias is created by accident in a sense, since it is due to the boundary of the domain of integration. One can see the dense sampling near τ == 0 up until t > 1 or so. To the right of t == 1, NIntegrate no longer samples so intensively.


On the other hand, when the dt integral is done first, the sampling points fall on lines where Ï„ is constant, including many along influential region near Ï„ == 0. This method produces an accurate estimate of the integral with relatively few sampling points.


I also looked into Szabolcs' observation that Method -> "LocalAdaptive" produced better results. This method refines the sampling in a different way than the default "GlobalAdaptive" method. It detects that sampling needs to be refined near Ï„ == 0, and produces a better estimate. It however does many more evaluations and takes much longer.


Finally I tried splitting the region up at Ï„ == 0 and t == 0. While it might be straying a bit from the OP's question, it seemed like a strategy appropriate to the integral. This produced the most accurate result (with other settings being the defaults) but did a lot of evaluations, although fewer than "LocalAdaptive".


(* Calculations *)


{tauval, {taupts}} = (* dτ first *)
Reap@NIntegrate[

integrand, {t, -∞, ∞}, {Ï„, -∞, t}, EvaluationMonitor :> Sow[{t, Ï„}]];

{tval, {tpts}} = (* dt first *)
Reap@NIntegrate[
integrand, {Ï„, -∞, ∞}, {t, Ï„, ∞}, EvaluationMonitor :> Sow[{t, Ï„}]];

{laval, {lapts}} =
Reap@NIntegrate[
integrand, {t, -∞, ∞}, {Ï„, -∞, t}, Method -> {"LocalAdaptive"},
EvaluationMonitor :> Sow[{t, Ï„}]];


{splitval, {splitpts}} = Reap[
NIntegrate[
integrand, {t, -∞, 0}, {Ï„, -∞, t},
EvaluationMonitor :> Sow[{t, Ï„}]] +
NIntegrate[
integrand, {t, 0, ∞}, {Ï„, -∞, 0, t}, (* the 0 divides the Ï„ interval *)
EvaluationMonitor :> Sow[{t, Ï„}]]
];


exact = Integrate[integrand, {t, -∞, ∞}, {Ï„, -∞, t}]

(* Results: the errors and the sampling points. *)


Length /@ {taupts, tpts, lapts, splitpts}
(*
{28421, 2856, 688884, 292987}
*)

{tauval, tval, laval, splitval} - exact
(*

{ 0.176832 - 0.140172 I,
3.44892*10^-9 + 3.34102*10^-8 I,
-4.20314*10^-8 - 2.38821*10^-8 I,
8.93826*10^-10 + 1.01682*10^-11 I}
*)

(* Sampling points (above) *)


plotrange = Max[{taupts, tpts, lapts, splitpts}];
ClearAll[showpts, showrow];
SetAttributes[showpts, HoldFirst];

showpts[pts_, opts___] :=
Graphics[{Red, PointSize[Tiny], Point[pts]}, opts, Frame -> True,
FrameLabel -> {t, Ï„}, PlotRange -> plotrange,
PlotRangePadding -> Scaled[0.03], PlotRangeClipping -> True,
PlotLabel -> (Hold[pts] /. {Hold[taupts] -> "dτ first", Hold[tpts] -> "dt first",
Hold[splitpts] -> "split dτ first", _ -> "LocalAdaptive"})];
SetAttributes[showrow, HoldFirst];
showrow[pts_] := showpts[pts, PlotRange -> #] & /@ {plotrange, plotrange/100, 2.5};

GraphicsGrid[{

showrow[taupts],
showrow[tpts],
showrow[lapts],
showrow[splitpts]
}]

Conclusion


We can see in the erroneous calculation (the dτ first NIntegrate), the sampling is not concentrated in all areas where the value of the integrand is significant and is evidently insufficient. For a full understanding, we would need to analyze the rule being used to estimate the integral, which I am not able to do. It is remarkable that the dt integral uses so few points (by comparison) to accurately estimate the integral, so it is likely that there is more to the story than just sampling. It is clearly the best way to numerically evaluate this integral. For instance the following produces produces an estimate that agrees with the exact result to 13 digits, takes only three times as long (1 sec.) and uses slightly fewer sample points than the machine precision calculation:


NIntegrate[integrand, {Ï„, -∞, ∞}, {t, Ï„, ∞}, WorkingPrecision -> 12]


As for how to know when a numerical result is inaccurate (to respond to a comment by the OP), I do not know of a foolproof way. Generally doing what the OP did, trying the calculation two different ways, is likely to catch an unstable computation. For instance, setting the WorkingPrecision to be 20 (I often choose a little more than machine precision), yields a warning and yet another, different value:


NIntegrate[integrand, {t, -∞, ∞}, {Ï„, -∞, t}, WorkingPrecision -> 20]


During evaluation of In[136]:= NIntegrate::slwcon: Numerical integration converging too slowly; suspect one of the following: singularity, value of the integration is 0, highly oscillatory integrand, or WorkingPrecision too small. >>



(*
0.06886985947874396770 + 0.42755314357642668589 I
*)


Sometimes just turning on precision tracking will help.


Comments

Popular posts from this blog

plotting - Plot 4D data with color as 4th dimension

I have a list of 4D data (x position, y position, amplitude, wavelength). I want to plot x, y, and amplitude on a 3D plot and have the color of the points correspond to the wavelength. I have seen many examples using functions to define color but my wavelength cannot be expressed by an analytic function. Is there a simple way to do this? Answer Here a another possible way to visualize 4D data: data = Flatten[Table[{x, y, x^2 + y^2, Sin[x - y]}, {x, -Pi, Pi,Pi/10}, {y,-Pi,Pi, Pi/10}], 1]; You can use the function Point along with VertexColors . Now the points are places using the first three elements and the color is determined by the fourth. In this case I used Hue, but you can use whatever you prefer. Graphics3D[ Point[data[[All, 1 ;; 3]], VertexColors -> Hue /@ data[[All, 4]]], Axes -> True, BoxRatios -> {1, 1, 1/GoldenRatio}]

plotting - Mathematica: 3D plot based on combined 2D graphs

I have several sigmoidal fits to 3 different datasets, with mean fit predictions plus the 95% confidence limits (not symmetrical around the mean) and the actual data. I would now like to show these different 2D plots projected in 3D as in but then using proper perspective. In the link here they give some solutions to combine the plots using isometric perspective, but I would like to use proper 3 point perspective. Any thoughts? Also any way to show the mean points per time point for each series plus or minus the standard error on the mean would be cool too, either using points+vertical bars, or using spheres plus tubes. Below are some test data and the fit function I am using. Note that I am working on a logit(proportion) scale and that the final vertical scale is Log10(percentage). (* some test data *) data = Table[Null, {i, 4}]; data[[1]] = {{1, -5.8}, {2, -5.4}, {3, -0.8}, {4, -0.2}, {5, 4.6}, {1, -6.4}, {2, -5.6}, {3, -0.7}, {4, 0.04}, {5, 1.0}, {1, -6.8}, {2, -4.7}, {3, -1....

functions - Get leading series expansion term?

Given a function f[x] , I would like to have a function leadingSeries that returns just the leading term in the series around x=0 . For example: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x)] x and leadingSeries[(1/x + 2 + (1 - 1/x^3)/4)/(4 + x)] -(1/(16 x^3)) Is there such a function in Mathematica? Or maybe one can implement it efficiently? EDIT I finally went with the following implementation, based on Carl Woll 's answer: lds[ex_,x_]:=( (ex/.x->(x+O[x]^2))/.SeriesData[U_,Z_,L_List,Mi_,Ma_,De_]:>SeriesData[U,Z,{L[[1]]},Mi,Mi+1,De]//Quiet//Normal) The advantage is, that this one also properly works with functions whose leading term is a constant: lds[Exp[x],x] 1 Answer Update 1 Updated to eliminate SeriesData and to not return additional terms Perhaps you could use: leadingSeries[expr_, x_] := Normal[expr /. x->(x+O[x]^2) /. a_List :> Take[a, 1]] Then for your examples: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x), x] leadingSeries[Exp[x], x] leadingSeries[(1/x + 2 + (1 - 1/x...