qml.fourier.qnode_spectrum¶
- qnode_spectrum(qnode, encoding_args=None, argnum=None, decimals=8, validation_kwargs=None)[source]¶
Compute the frequency spectrum of the Fourier representation of quantum circuits, including classical preprocessing.
The circuit must only use gates as input-encoding gates that can be decomposed into single-parameter gates of the form \(e^{-i x_j G}\) , which allows the computation of the spectrum by inspecting the gates’ generators \(G\). The most important example of such single-parameter gates are Pauli rotations.
The argument
argnum
controls which QNode arguments are considered as encoded inputs and the spectrum is computed only for these arguments. The input-encoding gates are those that are controlled by input-encoding QNode arguments. If noargnum
is given, all QNode arguments are considered to be input-encoding arguments.Note
Arguments of the QNode or parameters within an array-valued QNode argument that do not contribute to the Fourier series of the QNode with any frequency are considered as contributing with a constant term. That is, a parameter that does not control any gate has the spectrum
[0]
.- Parameters
qnode (pennylane.QNode) –
QNode
to compute the spectrum forencoding_args (dict[str, list[tuple]], set) – Parameter index dictionary; keys are argument names, values are index tuples for that argument or an
Ellipsis
. If aset
, all values are set toEllipsis
. The contained argument and parameter indices indicate the scalar variables for which the spectrum is computedargnum (list[int]) – Numerical indices for arguments with respect to which to compute the spectrum
decimals (int) – number of decimals to which to round frequencies.
validation_kwargs (dict) – Keyword arguments passed to
is_independent()
when testing for linearity of classical preprocessing in the QNode.
- Returns
Function which accepts the same arguments as the QNode. When called, this function will return a dictionary of dictionaries containing the frequency spectra per QNode parameter.
- Return type
function
Details
A circuit that returns an expectation value of a Hermitian observable which depends on \(N\) scalar inputs \(x_j\) can be interpreted as a function \(f: \mathbb{R}^N \rightarrow \mathbb{R}\). This function can always be expressed by a Fourier-type sum
\[\sum \limits_{\omega_1\in \Omega_1} \dots \sum \limits_{\omega_N \in \Omega_N} c_{\omega_1,\dots, \omega_N} e^{-i x_1 \omega_1} \dots e^{-i x_N \omega_N}\]over the frequency spectra \(\Omega_j \subseteq \mathbb{R},\) \(j=1,\dots,N\). Each spectrum has the property that \(0 \in \Omega_j\), and the spectrum is symmetric (i.e., for every \(\omega \in \Omega_j\) we have that \(-\omega \in\Omega_j\)). If all frequencies are integer-valued, the Fourier sum becomes a Fourier series.
As shown in Vidal and Theis (2019) and Schuld, Sweke and Meyer (2020), if an input \(x_j, j = 1 \dots N\), only enters into single-parameter gates of the form \(e^{-i x_j G}\) (where \(G\) is a Hermitian generator), the frequency spectrum \(\Omega_j\) is fully determined by the eigenvalues of the generators \(G\). In many situations, the spectra are limited to a few frequencies only, which in turn limits the function class that the circuit can express.
The
qnode_spectrum
function computes all frequencies that will potentially appear in the sets \(\Omega_1\) to \(\Omega_N\).Note
The
qnode_spectrum
function also supports preprocessing of the QNode arguments before they are fed into the gates, as long as this processing is linear. In particular, constant prefactors for the encoding arguments are allowed.Warning
In order to validate the preprocessing of the QNode arguments, automatic differentiation is used by
qnode_spectrum
. Therefore, pure Numpy parameters are not supported, but one of the machine learning frameworks has to be used.Example
Consider the following example, which uses non-trainable inputs
x
,y
andz
as well as trainable parametersw
as arguments to the QNode.n_qubits = 3 dev = qml.device("default.qubit", wires=n_qubits) @qml.qnode(dev) def circuit(x, y, z, w): for i in range(n_qubits): qml.RX(0.5*x[i], wires=i) qml.Rot(w[0,i,0], w[0,i,1], w[0,i,2], wires=i) qml.RY(2.3*y[i], wires=i) qml.Rot(w[1,i,0], w[1,i,1], w[1,i,2], wires=i) qml.RX(z, wires=i) return qml.expval(qml.Z(0))
This circuit looks as follows:
>>> from pennylane import numpy as pnp >>> x = pnp.array([1., 2., 3.]) >>> y = pnp.array([0.1, 0.3, 0.5]) >>> z = pnp.array(-1.8) >>> w = pnp.random.random((2, n_qubits, 3)) >>> print(qml.draw(circuit)(x, y, z, w)) 0: ──RX(0.50)──Rot(0.09,0.46,0.54)──RY(0.23)──Rot(0.59,0.22,0.05)──RX(-1.80)─┤ <Z> 1: ──RX(1.00)──Rot(0.98,0.61,0.07)──RY(0.69)──Rot(0.62,0.00,0.28)──RX(-1.80)─┤ 2: ──RX(1.50)──Rot(0.65,0.07,0.36)──RY(1.15)──Rot(0.74,0.27,0.24)──RX(-1.80)─┤
Applying the
qnode_spectrum
function to the circuit for the non-trainable parameters, we obtain:>>> res = qml.fourier.qnode_spectrum(circuit, argnum=[0, 1, 2])(x, y, z, w) >>> for inp, freqs in res.items(): ... print(f"{inp}: {freqs}") "x": {(0,): [-0.5, 0.0, 0.5], (1,): [-0.5, 0.0, 0.5], (2,): [-0.5, 0.0, 0.5]} "y": {(0,): [-2.3, 0.0, 2.3], (1,): [-2.3, 0.0, 2.3], (2,): [-2.3, 0.0, 2.3]} "z": {(): [-3.0, -2.0, -1.0, 0.0, 1.0, 2.0, 3.0]}
Note
While the Fourier spectrum usually does not depend on trainable circuit parameters or the actual values of the inputs, it may still change based on inputs to the QNode that alter the architecture of the circuit.
Usage Details
Above, we selected all input-encoding parameters for the spectrum computation, using the
argnum
keyword argument. We may also restrict the full analysis to a single QNode argument, again usingargnum
:>>> res = qml.fourier.qnode_spectrum(circuit, argnum=[0])(x, y, z, w) >>> for inp, freqs in res.items(): ... print(f"{inp}: {freqs}") "x": {(0,): [-0.5, 0.0, 0.5], (1,): [-0.5, 0.0, 0.5], (2,): [-0.5, 0.0, 0.5]}
Selecting arguments by name instead of index is possible via the
encoding_args
argument:>>> res = qml.fourier.qnode_spectrum(circuit, encoding_args={"y"})(x, y, z, w) >>> for inp, freqs in res.items(): ... print(f"{inp}: {freqs}") "y": {(0,): [-2.3, 0.0, 2.3], (1,): [-2.3, 0.0, 2.3], (2,): [-2.3, 0.0, 2.3]}
Note that for array-valued arguments the spectrum for each element of the array is computed. A more fine-grained control is available by passing index tuples for the respective argument name in
encoding_args
:>>> encoding_args = {"y": [(0,),(2,)]} >>> res = qml.fourier.qnode_spectrum(circuit, encoding_args=encoding_args)(x, y, z, w) >>> for inp, freqs in res.items(): ... print(f"{inp}: {freqs}") "y": {(0,): [-2.3, 0.0, 2.3], (2,): [-2.3, 0.0, 2.3]}
Warning
The
qnode_spectrum
function checks whether the classical preprocessing between QNode and gate arguments is linear by computing the Jacobian of the processing and applyingis_independent()
. This makes it unlikely – but not impossible – that non-linear functions go undetected. The number of additional points at which the Jacobian is computed in the numerical test ofis_independent
as well as other options for this function can be controlled viavalidation_kwargs
. Furthermore, the QNode arguments not marked inargnum
will not be considered in this test and if they resemble encoded inputs, the entire spectrum might be incorrect or the circuit might not even admit one.The
qnode_spectrum
function works in all interfaces:import tensorflow as tf dev = qml.device("default.qubit", wires=1) @qml.qnode(dev, interface='tf') def circuit(x): qml.RX(0.4*x[0], wires=0) qml.PhaseShift(x[1]*np.pi, wires=0) return qml.expval(qml.Z(0)) x = tf.Variable([1., 2.]) res = qml.fourier.qnode_spectrum(circuit)(x)
>>> print(res) {"x": {(0,): [-0.4, 0.0, 0.4], (1,): [-3.14159, 0.0, 3.14159]}}
Finally, compare
qnode_spectrum
withcircuit_spectrum()
, using the following circuit.dev = qml.device("default.qubit", wires=2) @qml.qnode(dev) def circuit(x, y, z): qml.RX(0.5*x**2, wires=0, id="x") qml.RY(2.3*y, wires=1, id="y0") qml.CNOT(wires=[1,0]) qml.RY(z, wires=0, id="y1") return qml.expval(qml.Z(0))
First, note that we assigned
id
labels to the gates for which we will usecircuit_spectrum
. This allows us to choose these gates in the computation:>>> x, y, z = pnp.array(0.1, 0.2, 0.3) >>> circuit_spec_fn = qml.fourier.circuit_spectrum(circuit, encoding_gates=["x","y0","y1"]) >>> circuit_spec = circuit_spec_fn(x, y, z) >>> for _id, spec in circuit_spec.items(): ... print(f"{_id}: {spec}") x: [-1.0, 0, 1.0] y0: [-1.0, 0, 1.0] y1: [-1.0, 0, 1.0]
As we can see, the preprocessing in the QNode is not included in the simple spectrum. In contrast, the output of
qnode_spectrum
is:>>> adv_spec = qml.fourier.qnode_spectrum(circuit, encoding_args={"y", "z"})(x, y, z) >>> for _id, spec in adv_spec.items(): ... print(f"{_id}: {spec}") y: {(): [-2.3, 0.0, 2.3]} z: {(): [-1.0, 0.0, 1.0]}
Note that the values of the output are dictionaries instead of the spectrum lists, that they include the prefactors introduced by classical preprocessing, and that we would not be able to compute the advanced spectrum for
x
because it is preprocessed non-linearly in the gateqml.RX(0.5*x**2, wires=0, id="x")
.