Skip to main content

How to realize a Fourier Transform on a non-uniform sampling data


I have a non-uniform sampling data (in time domain) from a Michelson interference experiment, as shown in Fig 1. But it was not evenly sampled (the step length was not uniform), because of the imperfection in the experiment. Further, if we enlarge the figure, we can see some bad-sampled points in Fig. 2 (the plot of a[[3800 ;; 4300]]).


enter image description here


The horizontal axis is time in femto-second (10^-15 second). In frequency domain, the central wavelength is [Lambda] = 800 nm, and the corresponding central frequency should be c/[Lambda] = 3*10^8/(800*10^-9) = 3.75*(10^14) Hz.


    SetDirectory[NotebookDirectory[]];
name = "try.txt";
a = Import[name, "Table"];
ListLinePlot[a, PlotRange -> All, Joined -> False, Mesh -> Full, Axes -> False, Frame -> True, AspectRatio -> .75, LabelStyle -> Directive[Black, FontSize -> 18], FrameLabel -> {{"Amplitude", None}, {"Time (fs)", None}}, ImageSize -> {400, 300}]


My question is : how to carry out a Fourier Transform on this time - domain data, so as to obtain the spectral distribution. Any suggestion or help will be highly appreciated. Thank you all in advance.



Answer



You can do an "irregular" Fourier transform by accounting for explicit horizontal axis values.


IFT[(w_)?NumberQ, vals_, times_] := 
Total[vals*Exp[(-I)*w*times]]/Length[times]

As coded above this is slow but there are sampling theorem based methods that bring it closer to the FFT in terms of speed.


I removed the HTML parts and put the file "try.txt" into my /tmp directory.


data = Import["/tmp/try.txt", "Data"];

Dimensions[data]
{times, vals} = Transpose[data];

(* Out[1015]= {7469, 2} *)

First an overall picture.


Plot[Abs[IFT[w, vals, times]], {w, 0, 202}, PlotRange -> All]

enter image description here


Besides what appears to be a DC component (hard to see since it is blue on black at the y axis), there seems to be a spikes located at a bit under 50, with smaller ones at multiples thereof. So we'll home in on that fundamental harmonic.



Plot[Abs[IFT[w, vals, times]], {w, 43, 52}, PlotRange -> All]

enter image description here


--- edit ---


Given the confusion (especially mine) over what is or might be the fundamental frequency, I did some more tests. The results might be of interest.


First I will remark that at frequency w=.2 (using the Exp[2*Pi*I*w...] normalization) I only get a small spike as compared to the larger one around 7.5. It's certainly noticeable in its neighborhood, but it's a very low-lying neighborhood.


Here is a good utility function. Given a putative period, it folds time values modulo that period. We will use it in ListPlot.


foldTimes[times_, period_] := period*FractionalPart[times/period]

I tried several values in the area of the peaks my IFT gave near .2 and the one below seemed to be among the more promising in terms of visibly showing something "periodic".



ListPlot[Transpose[{foldTimes[times, 1/(.209)], vals}]]

enter image description here


I will point out that using .204 instead of .209 gives a very different picture, but still one that looks "periodic". So I think we may have multiple and possibly incommensurate periods floating around. That said, I confess this visual analysis of signals is not a strong area for me.


Again with IFT redefined as below, we get a spike at 7.508, so we try the folded list plot using that reciprocal as putative period.


IFT[(w_)?NumberQ, vals_, times_] := 
Total[vals*Exp[(2*Pi*I)*w*times]]/Length[times]

ListPlot[Transpose[{foldTimes[times, 1/7.508], vals}]]


enter image description here


So that period would seem to have separate the lefties from the righties. But why did we get lefties and righties in this signal? I have no idea (maybe it's an election year phenomenon). Anyway, thought it was sufficiently pronounced as to warrant mention and a picture.


--- end edit ---


--- edit 2 ---


Ack, I think all I did was approximate the "granularity" of the time steps. So it was all noise, literally and figuratively I guess. The smaller peak around .2 is real enough though.


--- end edit 2 ---


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