Skip to content

Quantum models

A quantum program can be expressed and executed using the QuantumModel type. It serves three primary purposes:

Parameter handling: by conveniently handling and embedding the two parameter types that Qadence supports: feature and variational (see more details in the previous section).

Differentiability: by enabling a differentiable backend that supports two differentiable modes: automatic differentiation (AD) and parameter shift rules (PSR). The former is used general differentiation in statevector simulators based on PyTorch and JAX. The latter is a quantum specific method used to differentiate gate parameters, and is enabled for all backends.

Execution: by defining which backend the program is expected to be executed on. Qadence supports circuit compilation to the native backend representation.

Backends

The goal is for quantum models to be executed seemlessly on a number of different purpose backends: simulators, emulators or real hardware. By default, Qadence executes on the PyQTorch backend which implements a state vector simulator. Currently, this is the most feature rich backend. The Pulser backend is being developed, and currently supports a more limited set of functionalities (pulse sequences on programmable neutral atom arrays). The Horqrux backend, built on JAX, is also available, but currently not supported with the QuantumModel interface. For more information see the backend section.

The base QuantumModel exposes the following methods:

  • QuantumModel.run(): To extract the wavefunction after circuit execution. Not supported by all backends.
  • QuantumModel.sample(): Sample a bitstring from the resulting quantum state after circuit execution. Supported by all backends.
  • QuantumModel.expectation(): Compute the expectation value of an observable.

Every QuantumModel is an instance of a torch.nn.Module that enables differentiability for its expectation method. For statevector simulators, AD also works for the statevector itself.

To construct a QuantumModel, the program block must first be initialized into a QuantumCircuit instance by combining it with a Register. An integer number can also be passed for the total number of qubits, which instantiates a Register automatically. The qubit register also includes topological information on the qubit layout, essential for digital-analog computations. However, we will explore that in a later tutorial. For now, let's construct a simple parametrized quantum circuit.

from qadence import QuantumCircuit, RX, RY, chain, kron
from qadence import FeatureParameter, VariationalParameter

theta = VariationalParameter("theta")
phi = FeatureParameter("phi")

block = chain(
    kron(RX(0, theta), RY(1, theta)),
    kron(RX(0, phi), RY(1, phi)),
)

circuit = QuantumCircuit(2, block)
unique_params = circuit.unique_parameters
unique_params = [theta, phi]

The model can then be instantiated. Similarly to the direct execution functions shown in the previous tutorial, the run, sample and expectation methods are available directly from the model.

import torch
from qadence import QuantumModel, PI, Z

observable = Z(0) + Z(1)

model = QuantumModel(circuit, observable)

values = {"phi": torch.tensor([PI, PI/2])}

wf = model.run(values)
xs = model.sample(values, n_shots=100)
ex = model.expectation(values)
wf = tensor([[ 0.0690+0.0000j, -0.2535+0.0000j,  0.0000+0.2535j,  0.0000-0.9310j],
        [ 0.2465+0.0000j,  0.4310+0.0000j,  0.0000-0.4310j,  0.0000-0.7535j]])
xs = [OrderedCounter({'11': 89, '10': 6, '01': 4, '00': 1}), OrderedCounter({'11': 50, '10': 25, '01': 16, '00': 9})]
ex = tensor([[-1.7239],
        [-1.0139]])

By default, the forward method of QuantumModel calls model.run(). To define custom quantum models, the best way is to inherit from QuantumModel and override the forward method, as typically done with custom PyTorch Modules.

The QuantumModel class provides convenience methods to manipulate parameters. Being a torch.nn.Module, all torch methods are also available.

# To pass onto a torch optimizer
parameter_generator = model.parameters()

# Number of variational parameters
num_vparams = model.num_vparams

# Dictionary to easily inspect variational parameter values
vparams_values = model.vparams
vparams_values = OrderedDict([('theta', tensor([0.5317]))])

Model output

The output of a quantum model is typically encoded in the measurement of an expectation value. In Qadence, one way to customize the number of outputs is by batching the number of observables at model creation by passing a list of blocks.

from torch import tensor
from qadence import chain, kron, VariationalParameter, FeatureParameter
from qadence import QuantumModel, QuantumCircuit, PI, Z, RX, CNOT

theta = VariationalParameter("theta")
phi = FeatureParameter("phi")

block = chain(
    kron(RX(0, phi), RX(1, phi)),
    CNOT(0, 1)
)

circuit = QuantumCircuit(2, block)

model = QuantumModel(circuit, [Z(0), Z(0) + Z(1)])

values = {"phi": tensor(PI)}

ex = model.expectation(values)
ex = tensor([[-1.0000e+00, -7.4988e-33]])

As mentioned in the previous tutorial, blocks can also be arbitrarily parameterized through multiplication, which allows the inclusion of trainable parameters in the definition of the observable.

from qadence import I, Z

a = VariationalParameter("a")
b = VariationalParameter("b")

# Magnetization with a trainable shift and scale
observable = a * I(0) + b * Z(0)

model = QuantumModel(circuit, observable)

Quantum Neural Network (QNN)

The QNN is a subclass of the QuantumModel geared towards quantum machine learning and parameter optimisation. See the quantum machine learning section section or the QNN API reference for more detailed information. There are three main differences in interface when compared with the QuantumModel:

  • It is initialized with a list of the input parameter names, and then supports direct torch.Tensor inputs instead of the values dictionary shown above. The ordering of the input values should respect the order given in the input names.
  • Passing an observable is mandatory.
  • The forward method calls model.expectation().
from torch import tensor
from qadence import chain, kron, VariationalParameter, FeatureParameter
from qadence import QNN, QuantumCircuit, PI, Z, RX, RY, CNOT

theta = FeatureParameter("theta")
phi = FeatureParameter("phi")

block = chain(
    kron(RX(0, phi), RX(1, phi)),
    kron(RY(0, theta), RY(1, theta)),
    CNOT(0, 1)
)

circuit = QuantumCircuit(2, block)
observable = Z(0) + Z(1)

model = QNN(circuit, observable, inputs = ["phi", "theta"])

# "phi" = PI, PI/2, "theta" = 0.0, 1.0
values = tensor([[PI, 0.0], [PI/2, 1.0]])

ex = model(values)
ex = tensor([[-7.4988e-33],
        [ 1.1102e-16]])