Translation symmetry in 1D with OperatorTS1D

Here we will show how to take advantage of translation symmetry to save time and memory in PauliStrings.jl. Consider the 1D Ising Hamiltonian with periodic boundary conditions $H=-J(\sum_{i}Z_i Z_{i+1} +g \sum_i X_i)$. There is no need to actually store all the Pauli strings in this case. H is fully specified by just $-JZ_1Z_2$ and $-Jg X_1$ and the fact that it's translation symmetric.

In general, a 1D translation symmetric operator can be written as $\sum_i T_i H_0$ where $T_i$ is the i-sites translation operator and $H_0$ is the local operator that generates $H$. $H_0$ can be chosen so that it's only composed of Pauli strings that start on the first site.

In PauliStrings.jl, the structure OperatorTS1D lets you manipulate operators in this format. If your problem is 1D translation symmetric, OperatorTS1D will be much faster than Operator.

Construction

There are two ways of constructing an OperatorTS1D :

From the full translation symmetric $H$

First we construct the full Operator:

function ising1D(N, J, g)
    H = Operator(N)
    for j in 1:(N - 1)
        H += "Z",j,"Z",j+1
    end
    H += "Z",1,"Z",N #periodic boundary condition
    for j in 1:N
        H += g,"X",j
    end
    return -J*H
end
H = ising1D(N, J, g)

then convert it to an OperatorTS1D

Hts = OperatorTS1D(H)

If H is not translation symmetric, then an error will be returned

From the local generator $H_0$

Construct $H_0$ using Operator:

H = Operator(N)
H += -J, "Z",1,"Z",2
H += -J*g,"X",1

then convert it to an OperatorTS1D

Hts = OperatorTS1D(H, full=false)

Note that here, we need to set full=false in order to specify that we are not passing the full Hamiltonian but just its local generator.

Manipulation

All the operations defined on Operator are also defined on OperatorTS1D.

Construct a simple translation invariant operator on 4 sites:

H = Operator(N)
H += "X", 1
H += "Z", 1,"Z", 2

Hts = OperatorTS1D(H, full=false)

println(H)
println(Operator(Hts))
(1.0 + 0.0im) X111
(1.0 + 0.0im) ZZ11

(1.0 + 0.0im) 1ZZ1
(1.0 + 0.0im) 1X11
(1.0 + 0.0im) X111
(1.0 + 0.0im) Z11Z
(1.0 + 0.0im) 11ZZ
(1.0 + 0.0im) 11X1
(1.0 + 0.0im) ZZ11
(1.0 + 0.0im) 111X

Note that only the local generator is printed when printing OperatorTS1D, not the full operator.

Multiplication:

julia> Hts*Hts
(1.0 + 0.0im) X1X1
(2.0 + 0.0im) Z1Z1
(2.0 + 0.0im) 1111
(1.0 + 0.0im) ZZZZ
(2.0 + 0.0im) XZZ1
(2.0 + 0.0im) XX11
(2.0 + 0.0im) ZZX1

Addition:

julia> Hts+Hts
(2.0 + 0.0im) X111
(2.0 + 0.0im) ZZ11

etc.

Example: computing $Tr(H^k)$

As an example of performance gains of using OperatorTS1D instead of Operator we compute the 8th moment ($Tr(H^8)$) of a 30 spin system.

Using the function defined above we construct a Ising Hamiltonian and convert it to an OperatorTS1D:

N = 30
k = 8
H = ising1D(N, 1)
Hts = OperatorTS1D(H)

then we compute the kth moment (trace_product) using both Operator and OperatorTS1D :

julia> @time println(trace_product(H, k))
1.1904927790006272e18 + 0.0im
 80.697013 seconds (28.91 k allocations: 111.213 MiB, 0.07% gc time, 0.04% compilation time)

julia> @time println(trace_product(Hts, k))
1.190492779000627e18 + 0.0im
  1.951678 seconds (37.09 k allocations: 36.165 MiB, 2.00% gc time, 2.01% compilation time)

OperatorTS1D is 40 times faster in this case.