Skip to main content

probability or statistics - Radial distribution function (pair correlation function) for particles in rectangular image


Some time ago I asked the following question on how to calculate the radial distribution function.


I have here an add-on question, but before asking it I try to summarize and visualize my problem.




As input I have an image where particles are seen as bright dots:


enter image description here



After determining the positions of all particles, by e.g following code:


pts = ComponentMeasurements[Binarize@ImageSubtract
[image, BilateralFilter[image, 4, 1]], "Centroid"];

the next step is to calculate the radial distribution function:



The radial density distribution counts the number of points in a distance between $r$ and $r +\Delta r$ from each considered central point (below one marked as red). The area of such a "shell" is $2\pi r \Delta r$. The density distributions are averaged for all center points and then normalized by the total point density times the ring area for each radius.



enter image description here




It is important that the maximum radius (the maximum shell) for each center point does not cross the edges of the available point coordiantes (corresponding to the range defined by the smallest and largest x and y point value). That means that a point close to the edges has a maller maximum radius (smallest distance to the next edge) than a center point (for a square: maximum radius = half of the diagonal).





RunnyKines solution was:


radialDistributionFunction2D[pts_?MatrixQ, boxLength_Real, nBins_: 350] :=
Module[{gr, r, binWidth = boxLength/(2 nBins), npts = Length@pts, rho},
rho = npts/boxLength^2; (* area number density *)
{r, gr} = HistogramList[(*compute and bin the distances between points of interest*)
Flatten @ DistanceMatrix @ pts, {0.005, boxLength/4., binWidth}];
r = MovingMedian[r, 2]; (* take center of each bin as r *)

gr = gr/(2 Pi r rho binWidth npts); (* normaliza g(r) *)
Transpose[{r, gr}] (* combine r and g(r) *)
]

rdf = radialDistributionFunction2D[pts, 1023.];
ListLinePlot[rdf, PlotRange ->{{0, 150}, All}, Mesh -> 80]

enter image description here




My question is:



How can I consider all points as center points. Particles in rings which cross the edge of the image should also be taken into acount. The maximum radius of each point is depending on its distance to the next edge of the image. That would improve the statistics for the small distances (less than a quarter of the whole image).



Answer



The range of the bins for your image is 0 to Sqrt[2] 1024 with bin size say binSz = 1/5.


If two points are uniformily distributed in the unit square, their distance has density:


pdf[z_] = Piecewise[{{2 z (Ï€ + (z - 4) z), z <= 1}, {4 z (2 Sqrt[z^2 - 1] - 1 - z^2/2 +
ArcCsc[z] - ArcTan[Sqrt[z^2 - 1]]), 1 < z < Sqrt[2]}}, 0]

Scaling the bins down by the image size and integrating the PDF over each scaled bin, we get the expected amount of distance matrix entries in each bin (relatively/unnormalized).


imSz = 1024;
binSz = 1/5;


expected = With[{cdf = Integrate[pdf[z], z],
jumps = Range[0, Ceiling[Sqrt[2], binSz/imSz], binSz/imSz]},
Differences[Table[cdf, {z, N[jumps, 20]}]]];

We obtain an unnormalized density by dividing the amount of distance matrix entries in each bin by expected.


counts = Counts[Quotient[Join @@ DistanceMatrix[pts], binSz]];
density = With[{quotients = Range[0, Floor[Sqrt[2] imSz/binSz]]},
Lookup[counts, quotients, 0]/expected];


(* median distance for each bin *)
dists = MovingAverage[Range[0, Ceiling[Sqrt[2] imSz, binSz], binSz], 2];

Show[ListLinePlot[#, PlotRange -> All, AspectRatio -> 1/3, PlotStyle -> Red],
ListPlot[#, PlotRange -> All, PlotStyle -> Directive[Black, AbsolutePointSize[3]]]] &[
Transpose[{dists, density}][[2 ;; #]]] & /@ {600, -150}


Given some distance, pdf[x/imSz] express how much data there will be on average. Its square root express the relative reliability of the estimate w.r.t. the distance.


Plot[Sqrt[pdf[x/imSz]], {x, 0, Sqrt[2] imSz}, AspectRatio -> 1/4]



Edit: In the modification below, the bin widths are chosen such that the reliability integrated over each bin is the same for all bins, because bigger bins are appropriate when the reliability is low.


ClearAll[Q]; (* Anti-derivative of the reliability *)
Q[z_] = Q[z] /. First[NDSolve[{Derivative[1][Q][z] == Sqrt[pdf[z]],
Q[0] == 0}, Q[z], {z, 0, Sqrt[2]}, InterpolationOrder -> All]];

binCount = 3500;
jumps = Module[{linear = Subdivide[0., Sqrt[2.], binCount],
pos, Qvals = Take[Subdivide[0., Q[Sqrt[2.]], binCount], {2, -2}]},

pos = Ordering[Ordering[Join[Q[linear], Qvals]], -#] - Range[#] &[Length[Qvals]];
Join[x /. MapThread[FindRoot[Q[x] - #, {x, 0.5 (#2 + #3), #2, #3}] &,
{Qvals, linear[[pos]], Rest[linear][[pos]]}], {Sqrt[2.], 0.}] // RotateRight];

expected = With[{cdf = Integrate[pdf[z], z]}, Differences[Table[cdf, {z, jumps}]]];

counts = Differences[Ordering[Ordering[Join[Drop[Catenate[DistanceMatrix[
Developer`ToPackedArray[pts]]], {1, -1, Length[pts] + 1}],
Developer`ToPackedArray[N[imSz jumps]]]], -Length[jumps]]] - 1;


(* bin medians and mass *)
dists = imSz MovingAverage[jumps, 2];
density = counts/expected;

With[{data = Transpose[{dists, density}]}, Show[
ListLinePlot[data, PlotRange -> All, AspectRatio -> 1/3, PlotStyle -> Red],
ListPlot[data, PlotRange -> All, PlotStyle -> Directive[Black, AbsolutePointSize[1.5]]],
PlotRange -> {{0.`, imSz Sqrt[2.]}, {0.`, 1.5*10^8}}]]



Comments

Popular posts from this blog

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

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

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