Ramsey Spectroscopy
A typical problem in quantum optics is the interrogation of an atom with an external driving field. In this brief example we apply the Ramsey interferometry on a single two-level atom. The distinctiveness of this method is the time dependent laser field. In particular, two short $\pi/2$-pulses are employed with a free evolution between them. This system is described by the simple Hamiltonian
\[H = - \Delta \sigma^{22} + \Omega(t) (\sigma^{21} + \sigma^{12}),\]
where $\Delta = \omega_l - \omega_a$ is the detuning between the laser and atomic resonance frequency, and $\Omega(t)$ describes the time dependent driving field. Additionally we include atomic decay and dephasing, with the rates $\Gamma$ and $\nu$, respectively.
We start by loading the packages.
using QuantumCumulants
using OrdinaryDiffEq, ModelingToolkit
using Plots
Beside defining the symbolic parameters we additionally need to register a time dependent external function $f(t)$ for our driving laser. To this end we first need to define the independent variable $t$ of our time evolution. Due to the registration, $f(t)$ is treated as a parameter in the symbolic equations, but at the numerical evaluation we can assign every function to it.
@cnumbers Δ Ω Γ ν
@syms t::Real
@register_symbolic f(t)
After defining the Hilbert space and the operator of the two-level atom we construct the time dependent Hamiltonian as well as the jump operator list with the corresponding rates.
h = NLevelSpace(:atom, 2) # Hilbert space
σ(i, j) = Transition(h, :σ, i, j, 2) # Operators
H = -Δ*σ(2, 2) + f(t)*Ω/2*(σ(1, 2) + σ(2, 1)) # Hamiltonian
J = [σ(1, 2), σ(2, 2)] # Jump operators & rates
rates = [Γ, ν]
The two-level atom is completely described with the excited state population and the coherence. Therefore, we derive the equations for these two operators.
eqs = meanfield([σ(2, 2), σ(1, 2)], H, J; rates = rates)
\[\begin{align} \frac{d}{dt} \langle {\sigma}^{{22}}\rangle =& -1.0 \Gamma \langle {\sigma}^{{22}}\rangle + \frac{1}{2} i \Omega f\left( t \right) \langle {\sigma}^{{12}}\rangle + \frac{-1}{2} i \Omega f\left( t \right) \langle {\sigma}^{{21}}\rangle \\ \frac{d}{dt} \langle {\sigma}^{{12}}\rangle =& \frac{-1}{2} i \Omega f\left( t \right) -0.5 \Gamma \langle {\sigma}^{{12}}\rangle -0.5 \nu \langle {\sigma}^{{12}}\rangle + 1 i \Delta \langle {\sigma}^{{12}}\rangle + 1 i \Omega f\left( t \right) \langle {\sigma}^{{22}}\rangle \end{align}\]
To calculate the dynamic of the system we create a system of ordinary differential equations and define the numeric parameters with the time dependent function.
@named sys = System(eqs)
\[ \begin{align} \frac{\mathrm{d} \mathtt{\langle{\sigma}22\rangle}\left( t \right)}{\mathrm{d}t} &= - \mathtt{\langle{\sigma}22\rangle}\left( t \right) \Gamma + \frac{-1}{2}\mathit{i} conj\left( \mathtt{\langle{\sigma}12\rangle}\left( t \right) \right) f\left( t \right) \Omega + \frac{1}{2}\mathit{i} f\left( t \right) \mathtt{\langle{\sigma}12\rangle}\left( t \right) \Omega \\\\ \frac{\mathrm{d} \mathtt{\langle{\sigma}12\rangle}\left( t \right)}{\mathrm{d}t} &= \frac{-1}{2}\mathit{i} f\left( t \right) \Omega - 0.5 \mathtt{\langle{\sigma}12\rangle}\left( t \right) \left( \Gamma + \nu \right) + \mathit{i} \mathtt{\langle{\sigma}12\rangle}\left( t \right) \Delta + \mathit{i} f\left( t \right) \mathtt{\langle{\sigma}22\rangle}\left( t \right) \Omega \end{align} \]
Parameter
Γ_ = 1.0
Ω_ = 500Γ_
Δ_ = 0Γ_
ν_ = 0.2Γ_
tp = π/2Ω_ # π/2-pulse
tf = 1/20Γ_ # free evolution without drive
function f(t)
if t<tp || (t>tp+tf && t<2tp+tf)
return 1
else
0
end
end
ps = [Γ, Ω, Δ, ν]
p0 = [Γ_, Ω_, Δ_, ν_]
u0 = zeros(ComplexF64, length(eqs)) # Initial state
Finally, we calculate and plot the time evolution.
dict = merge(Dict(ps .=> p0), Dict(unknowns(sys) .=> u0))
prob = ODEProblem(sys, dict, (0.0, 2tp+tf))
sol = solve(prob, Tsit5(), maxiters = 1e7)
t = sol.t # Plot time evolution
s22 = real.(sol[σ(2, 2)])
plot(t, s22, xlabel = "tΓ", ylabel = "⟨σ22⟩", legend = false, size = (600, 300))
Scanning over the detuning for the excited state population leads to the well-known Ramsey fringes.
Δ_ls = [-2000:4:2000;]Γ_
s22_ls = zeros(length(Δ_ls))
for i = 1:length(Δ_ls)
dict = merge(Dict(ps .=> [Γ_; Ω_; Δ_ls[i]; ν_]), Dict(unknowns(sys) .=> u0))
prob_i = ODEProblem(sys, dict, (0.0, 2tp+tf))
sol_i = solve(prob_i, RK4(); adaptive = false, dt = 1e-5)
s22_ls[i] = real.(sol_i[σ(2, 2)])[end]
end
plot(Δ_ls, s22_ls, xlabel = "Δ/Γ", ylabel = "⟨σ22⟩", legend = false, size = (600, 300))
Package versions
These results were obtained using the following versions:
using InteractiveUtils
versioninfo()
using Pkg
Pkg.status(["QuantumCumulants", "OrdinaryDiffEq"], mode = PKGMODE_MANIFEST)
Julia Version 1.11.7
Commit f2b3dbda30a (2025-09-08 12:10 UTC)
Build Info:
Official https://julialang.org/ release
Platform Info:
OS: Linux (x86_64-linux-gnu)
CPU: 4 × AMD EPYC 7763 64-Core Processor
WORD_SIZE: 64
LLVM: libLLVM-16.0.6 (ORCJIT, znver3)
Threads: 4 default, 0 interactive, 2 GC (on 4 virtual cores)
Environment:
JULIA_PKG_SERVER_REGISTRY_PREFERENCE = eager
JULIA_DEBUG = Documenter
JULIA_NUM_THREADS = auto
Status `~/work/QuantumCumulants.jl/QuantumCumulants.jl/docs/Manifest.toml`
⌃ [1dea7af3] OrdinaryDiffEq v6.101.0
[35bcea6d] QuantumCumulants v0.4.2 `~/work/QuantumCumulants.jl/QuantumCumulants.jl`
Info Packages marked with ⌃ have new versions available and may be upgradable.
This page was generated using Literate.jl.