Quantum Circuit Compilation
Quantum Circuit Compilation is a technique that aims to reduce the depth of certain quantum circuits by approximating them by fixed depth variational circuits. A well-known example of this is quantum assisted quantum compiling (QAQC). This is available in QURI Algo through our compilation module, which will be introduced through this notebook.
Overview
In this notebook we will go through
- Some background of QAQC
- How to use the Hilbert-Schmidt cost function
- How to use the QURI Algo implementation of QAQC
QAQC
The objective of QAQC is to approximate a known quantum circuit which is understood to be too deep by a low-depth variational circuit . Ideally the depth of is or , which is achievable with either brick-wall or tensor-network inspired ansatze. A cost-function is needed, that when minimized ensures that for some global phase .
We can use the Hilbert-Schmidt (HS) test for this which is
The advantage of this is that it can be evaluated on quantum hardware by sampling a qubit wavefunction. This has implications for QPE where multi-qubit controlled time-evolution is performed several times. Although the requirement likely makes the optimization untractable using classical computing, running the optimization using a quantum device requires little additional overhead compared to QPE and has the potential to reduce the run-time of QPE dramatically.
To summarize it briefly, the estimation is done on a qubit quantum register consisting of -qubit subsystems and . The system has initially been prepared as a product of Bell states defined by pairs of qubits belonging to and . Denoting by and the 'th qubit on subsystem and respectively. We write
Then the initial state is
Then and are applied to the and subsystem respectively. The resulting state density matrix is
Then measurements are conducted in the basis of pair-wise Bell states . Resultingly, this gives us the HS test because
Here the operators are
Another variant is the local HS (LHS) test, which is defined as
with
We provide the hybrid cost-function
When running either the HS or LHS tests the measurements required are the same. For this reason, whenever a series of measurements are performed that result in the HS test, those same measurements if averaged over single Bell pairs could be used to simultaneously compute the LHS test. Therefore the above cost-function is no more expensive than either of its parts. There are certain situations where the LHS results in an optimization landscape that is easier for gradient based optimizers to navigate. This is because the inner product between and some randomly chosen is exponentially suppressed due to the scaling of the Hilbert space. The LHS test avoids this issue by weighting the local cost of each pair of qubits equally. So for large systems it is worthwhile to use .
Cost function
In this section we will show how to use the HS test. Although the objective of this notebook is to use it for circuit compilation, it is a rather versatile tool, and so when introducing it we will use it to investigate the Trotter time-evolution circuit. By the end of this section we will have experimented with several different Trotter steps and picked one that is faithful to the exact dynamics of our system.
First it needs to be imported from the cost_functions
module. It needs to be initialized with a quantum compilation backend that will take care of the estimation functions for us. This will use an estimator from QURI Parts.
from quri_parts.qulacs.estimator import create_qulacs_vector_estimator
from quri_algo.core.cost_functions import HilbertSchmidtTest
hstest = HilbertSchmidtTest(
create_qulacs_vector_estimator(), alpha=1.0
) # alpha=1.0 is the default.
Let's test it out.
import numpy as np
from quri_parts.circuit import QuantumCircuit
test_0 = QuantumCircuit(2)
test_0.add_RZ_gate(0, np.pi)
test_0.add_RX_gate(1, np.pi)
test_1 = QuantumCircuit(2)
test_1.add_RZ_gate(0, 0.5 * np.pi)
test_1.add_RX_gate(1, 0.5 * np.pi)
result = hstest(
test_0, test_1, alpha=0.7
) # When calling hstest we can supply an alpha that's different from the one we initialized.
print("The result from HS test is", result)
The result from HS test is _Estimate(value=(0.675+0j), error=0.0)
We can use a time-evolution circuit factory and generate a Trotterized time evolution based on a simple Hamiltonian. Here we pick the Heisenberg interaction on a spin-1/2 lattice with periodic boundary conditions. We will compute and plot
with being either the exact time-evolution operator or the one generated by Trotterization.
import numpy as np
import matplotlib.pyplot as plt
from numpy import linspace
from quri_parts.core.operator import Operator, pauli_label
from quri_algo.circuit.time_evolution.trotter_time_evo import (
TrotterTimeEvolutionCircuitFactory,
)
from quri_algo.circuit.time_evolution.exact_unitary import (
ExactUnitaryTimeEvolutionCircuitFactory,
)
from quri_algo.problem.hamiltonian import QubitHamiltonianInput
s = 1 / 2
j = 1.0
heisenberg = Operator(
{
pauli_label("X0 X1"): j * s**2,
pauli_label("X1 X2"): j * s**2,
pauli_label("X2 X3"): j * s**2,
pauli_label("X3 X0"): j * s**2,
pauli_label("Y0 Y1"): j * s**2,
pauli_label("Y1 Y2"): j * s**2,
pauli_label("Y2 Y3"): j * s**2,
pauli_label("Y3 Y0"): j * s**2,
pauli_label("Z0 Z1"): j * s**2,
pauli_label("Z1 Z2"): j * s**2,
pauli_label("Z2 Z3"): j * s**2,
pauli_label("Z3 Z0"): j * s**2,
}
)
heisenberg_input = QubitHamiltonianInput(4, heisenberg)
circuit_factory = TrotterTimeEvolutionCircuitFactory(heisenberg_input, 1)
exact_time_evolution_factory = ExactUnitaryTimeEvolutionCircuitFactory(heisenberg_input)
start_circuit = circuit_factory(0.0)
evo_times = linspace(0, 4 * np.pi, 51)
hs_result_trotter = [
hstest(start_circuit, circuit_factory(time), alpha=0.5).value.real
for time in evo_times
]
hs_result_exact = [
hstest(start_circuit, exact_time_evolution_factory(time), alpha=0.5).value.real
for time in evo_times
]
ax = plt.axes()
ax.plot(evo_times, hs_result_trotter)
ax.plot(evo_times, hs_result_exact)
ax.set_title("Hilbert Schmidt test")
ax.set_xlabel("t")
ax.set_ylabel("C(U(0),U(t))")
ax.set_ybound(0.0, 1.05)
ax.legend(["Trotter", "Exact"])