Source code for pennylane.pulse.parametrized_evolution

# Copyright 2018-2023 Xanadu Quantum Technologies Inc.

# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at

#     http://www.apache.org/licenses/LICENSE-2.0

# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

# pylint: disable=too-few-public-methods,function-redefined

"""
This file contains the ``ParametrizedEvolution`` operator.
"""

import warnings
from collections.abc import Sequence
from typing import Union

import pennylane as qml
from pennylane.operation import AnyWires, Operation
from pennylane.ops import functions
from pennylane.typing import TensorLike

from .hardware_hamiltonian import HardwareHamiltonian
from .parametrized_hamiltonian import ParametrizedHamiltonian

has_jax = True
try:
    import jax
    import jax.numpy as jnp
    from jax.experimental.ode import odeint

    from .parametrized_hamiltonian_pytree import ParametrizedHamiltonianPytree
except ImportError as e:
    has_jax = False


[docs]class ParametrizedEvolution(Operation): r""" ParametrizedEvolution(H, params=None, t=None, return_intermediate=False, complementary=False, id=None, **odeint_kwargs) Parametrized evolution gate, created by passing a :class:`~.ParametrizedHamiltonian` to the :func:`~.pennylane.evolve` function For a time-dependent Hamiltonian of the form .. math:: H(\{v_j\}, t) = H_\text{drift} + \sum_j f_j(v_j, t) H_j it implements the corresponding time-evolution operator :math:`U(t_0, t_1)`, which is the solution to the time-dependent Schrodinger equation. .. math:: \frac{d}{dt}U(t) = -i H(\{v_j\}, t) U(t). Under the hood, it is using a numerical ordinary differential equation (ODE) solver. It requires ``jax``, and will not work with other machine learning frameworks typically encountered in PennyLane. Args: H (ParametrizedHamiltonian): Hamiltonian to evolve params (Optional[list]): trainable parameters, passed as list where each element corresponds to the parameters of a scalar-valued function of the Hamiltonian being evolved. t (Union[float, List[float]]): If a float, it corresponds to the duration of the evolution. If a list of floats, the ODE solver will use all the provided time values, and perform intermediate steps if necessary. It is recommended to just provide a start and end time unless matrices of the time evolution at intermediate times need to be computed. Note that such absolute times only have meaning within an instance of ``ParametrizedEvolution`` and will not affect other gates. To return the matrix at intermediate evolution times, activate ``return_intermediate`` (see below). id (str or None): id for the scalar product operator. Default is None. Keyword Args: atol (float, optional): Absolute error tolerance for the ODE solver. Defaults to ``1.4e-8``. rtol (float, optional): Relative error tolerance for the ODE solver. The error is estimated from comparing a 4th and 5th order Runge-Kutta step in the Dopri5 algorithm. This error is guaranteed to stay below ``tol = atol + rtol * abs(y)`` through adaptive step size selection. Defaults to 1.4e-8. mxstep (int, optional): maximum number of steps to take for each timepoint for the ODE solver. Defaults to ``jnp.inf``. hmax (float, optional): maximum step size allowed for the ODE solver. Defaults to ``jnp.inf``. return_intermediate (bool): Whether or not the ``matrix`` method returns all intermediate solutions of the time evolution at the times provided in ``t = [t_0,...,t_f]``. If ``False`` (the default), only the matrix for the full time evolution is returned. If ``True``, all solutions including the initial condition are returned; when used in a circuit, this results in ``ParametrizedEvolution`` being a broadcasted operation, see the usage details ("Computing intermediate time evolution") below. complementary (bool): Whether or not to compute the complementary time evolution when using ``return_intermediate=True`` (ignored otherwise). If ``False`` (the default), the usual solutions to the Schrodinger equation :math:`\{U(t_0, t_0), U(t_0, t_1),\dots, U(t_0, t_f)\}` are computed, where :math:`t_i` are the additional times provided in ``t``. If ``True``, the *remaining* time evolution to :math:`t_f` is computed instead, returning :math:`\{U(t_0, t_f), U(t_1, t_f),\dots, U(t_{f-1}, t_f), U(t_f, t_f)\}`. dense (bool): Whether the evolution should use dense matrices. Per default, this is decided by the number of wires, i.e. ``dense = len(wires) < 3``. .. warning:: The :class:`~.ParametrizedHamiltonian` must be Hermitian at all times. This is not explicitly checked when creating a :class:`~.ParametrizedEvolution` from the :class:`~.ParametrizedHamiltonian`. **Example** To create a :class:`~.ParametrizedEvolution`, we first define a :class:`~.ParametrizedHamiltonian` describing the system, and then pass it to :func:`~pennylane.evolve`: .. code-block:: python from jax import numpy as jnp f1 = lambda p, t: jnp.sin(p * t) H = f1 * qml.Y(0) ev = qml.evolve(H) The initial :class:`~.ParametrizedEvolution` does not have set parameters, and so will not have a matrix defined. To obtain an Operator with a matrix, it must be passed parameters and a time interval: >>> qml.matrix(ev([1.2], t=[0, 4])) Array([[ 0.72454906+0.j, -0.6892243 +0.j], [ 0.6892243 +0.j, 0.72454906+0.j]], dtype=complex64) The parameters can be updated by calling the :class:`~.ParametrizedEvolution` again with different inputs. When calling the :class:`~.ParametrizedEvolution`, keyword arguments can be passed to specify behaviour of the ODE solver. The :class:`~.ParametrizedEvolution` can be implemented in a QNode: .. code-block:: python import jax jax.config.update("jax_enable_x64", True) dev = qml.device("default.qubit", wires=1) @jax.jit @qml.qnode(dev, interface="jax") def circuit(params): qml.evolve(H)(params, t=[0, 10]) return qml.expval(qml.Z(0)) >>> params = [1.2] >>> circuit(params) Array(0.96632722, dtype=float64) >>> jax.grad(circuit)(params) [Array(2.35694829, dtype=float64)] .. note:: In the example above, the decorator ``@jax.jit`` is used to compile this execution just-in-time. This means the first execution will typically take a little longer with the benefit that all following executions will be significantly faster, see the jax docs on jitting. JIT-compiling is optional, and one can remove the decorator when only single executions are of interest. .. warning:: The time argument ``t`` corresponds to the time window used to compute the scalar-valued functions present in the :class:`ParametrizedHamiltonian` class. Consequently, executing two ``ParametrizedEvolution`` operators using the same time window does not mean both operators are executed simultaneously, but rather that both evaluate their respective scalar-valued functions using the same time window. See Usage Details. .. note:: Using ``return_intermediate`` in a quantum circuit leads to broadcasted execution, which can lead to unintended additional computational cost. Also consider the usage details below. .. details:: :title: Usage Details The parameters used when calling the ``ParametrizedEvolution`` are expected to have the same order as the functions used to define the :class:`~.ParametrizedHamiltonian`. For example: .. code-block:: python3 def f1(p, t): return jnp.sin(p[0] * t**2) + p[1] def f2(p, t): return p * jnp.cos(t) H = 2 * qml.X(0) + f1 * qml.Y(0) + f2 * qml.Z(0) ev = qml.evolve(H) >>> params = [[4.6, 2.3], 1.2] >>> qml.matrix(ev(params, t=0.5)) Array([[-0.18354285-0.26303384j, -0.7271658 -0.606923j ], [ 0.7271658 -0.606923j , -0.18354285+0.26303384j]], dtype=complex64) Internally the solver is using ``f1([4.6, 2.3], t)`` and ``f2(1.2, t)`` at each timestep when finding the matrix. In the case where we have defined two Hamiltonians, ``H1`` and ``H2``, and we want to find a time evolution where the two are driven simultaneously for some period of time, it is important that both are included in the same call of :func:`~.pennylane.evolve`. For non-commuting operations, applying ``qml.evolve(H1)(params, t=[0, 10])`` followed by ``qml.evolve(H2)(params, t=[0, 10])`` will **not** apply the two pulses simultaneously, despite the overlapping time window. Instead, it will execute ``H1`` in the ``[0, 10]`` time window, and then subsequently execute ``H2`` using the same time window to calculate the evolution, but without taking into account how the time evolution of ``H1`` affects the evolution of ``H2`` and vice versa. Consider two non-commuting :class:`ParametrizedHamiltonian` objects: .. code-block:: python from jax import numpy as jnp ops = [qml.X(0), qml.Y(1), qml.Z(2)] coeffs = [lambda p, t: p for _ in range(3)] H1 = qml.dot(coeffs, ops) # time-independent parametrized Hamiltonian ops = [qml.Z(0), qml.Y(1), qml.X(2)] coeffs = [lambda p, t: p * jnp.sin(t) for _ in range(3)] H2 = qml.dot(coeffs, ops) # time-dependent parametrized Hamiltonian The evolutions of the :class:`ParametrizedHamiltonian` can be used in a QNode. .. code-block:: python dev = qml.device("default.qubit", wires=3) @qml.qnode(dev, interface="jax") def circuit1(params): qml.evolve(H1)(params, t=[0, 10]) qml.evolve(H2)(params, t=[0, 10]) return qml.expval(qml.Z(0) @ qml.Z(1) @ qml.Z(2)) @qml.qnode(dev, interface="jax") def circuit2(params): qml.evolve(H1 + H2)(params, t=[0, 10]) return qml.expval(qml.Z(0) @ qml.Z(1) @ qml.Z(2)) In ``circuit1``, the two Hamiltonians are evolved over the same time window, but inside different operators. In ``circuit2``, we add the two to form a single :class:`~.ParametrizedHamiltonian`. This will combine the two so that the expected parameters will be ``params1 + params2`` (as an addition of ``list``). They can then be included inside a single :class:`~.ParametrizedEvolution`. The resulting evolutions of ``circuit1`` and ``circuit2`` are **not** identical: >>> params = jnp.array([1., 2., 3.]) >>> circuit1(params) Array(-0.01542578, dtype=float64) >>> params = jnp.concatenate([params, params]) # H1 + H2 requires 6 parameters! >>> circuit2(params) Array(-0.78235162, dtype=float64) Here, ``circuit1`` is not executing the evolution of ``H1`` and ``H2`` simultaneously, but rather executing ``H1`` in the ``[0, 10]`` time window and then executing ``H2`` with the same time window, without taking into account how the time evolution of ``H1`` affects the evolution of ``H2`` and vice versa! One can also provide a list of time values that the ODE solver will use to calculate the evolution of the ``ParametrizedHamiltonian``. Keep in mind that the ODE solver uses an adaptive step size, thus it might use additional intermediate time values. .. code-block:: python t = jnp.arange(0., 10.1, 0.1) @qml.qnode(dev, interface="jax") def circuit(params): qml.evolve(H1 + H2)(params, t=t) return qml.expval(qml.Z(0) @ qml.Z(1) @ qml.Z(2)) >>> circuit(params) Array(-0.78235162, dtype=float64) >>> jax.grad(circuit)(params) Array([-4.80708632, 3.70323783, -1.32958799, -2.40642477, 0.68105214, -0.52269657], dtype=float64) Given that we used the same time window (``[0, 10]``), the results are the same as before. **Computing intermediate time evolution** As discussed above, the ODE solver will evaluate the Schrodinger equation at intermediate times in any case. By passing additional time values explicitly in the time window ``t`` and setting ``return_intermediate=True``, the ``matrix`` method will return the matrices for the intermediate time evolutions as well: .. math:: \{U(t_0, t_0), U(t_0, t_1), \dots, U(t_0, t_{f-1}), U(t_0, t_f)\}. The first entry here is the initial condition :math:`U(t_0, t_0)=1`. For a simple time-dependent single-qubit Hamiltonian, this feature looks like the following: .. code-block:: python ops = [qml.Z(0), qml.Y(0), qml.X(0)] coeffs = [lambda p, t: p * jnp.cos(t) for _ in range(3)] H = qml.dot(coeffs, ops) # time-dependent parametrized Hamiltonian param = [jnp.array(0.2), jnp.array(1.1), jnp.array(-1.3)] time = jnp.linspace(0.1, 0.4, 6) # Six time points from 0.1 to 0.4 ev = qml.evolve(H)(param, time, return_intermediate=True) >>> ev_mats = ev.matrix() >>> ev_mats.shape (6, 2, 2) Note that the broadcasting axis has length ``len(time)`` and is the first axis of the returned tensor. We may use this feature within QNodes executed on a simulator, returning the measurements for all intermediate time steps: .. code-block:: python dev = qml.device("default.qubit", wires=1) @qml.qnode(dev, interface="jax") def circuit(param, time): qml.evolve(H)(param, time, return_intermediate=True) return qml.probs(wires=[0]) >>> circuit(param, time) Array([[1. , 0. ], [0.98977406, 0.01022594], [0.95990416, 0.04009584], [0.91236167, 0.08763833], [0.84996865, 0.15003133], [0.77614817, 0.22385181]], dtype=float64) **Computing complementary time evolution** When using ``return_intermediate=True``, the partial time evolutions share the *initial* time :math:`t_0`. For some applications, however, it may be useful to compute the complementary time evolutions, i.e. the partial evolutions that share the *final* time :math:`t_f`. This can be activated by setting ``complementary=True``, which will make ``ParametrizedEvolution.matrix`` return the matrices .. math:: \{U(t_0, t_f), U(t_1, t_f), \dots, U(t_f, t_f)\}. Using the Hamiltonian from the example above: >>> complementary_ev = ev(param, time, return_intermediate=True, complementary=True) >>> comp_ev_mats = complementary_ev.matrix() >>> comp_ev_mats.shape (6, 2, 2) If we multiply the matrices computed before with ``complementary=False`` with these complementary evolution matrices from the left, we obtain the full time evolution, which we can check by comparing to the last entry of ``ev_mats``: >>> for mat, c_mat in zip(ev_mats, comp_ev_mats): ... print(qml.math.allclose(c_mat @ mat, ev_mats[-1])) True True True True True True """ _name = "ParametrizedEvolution" num_wires = AnyWires grad_method = "A" # pylint: disable=too-many-arguments def __init__( self, H: ParametrizedHamiltonian, params: list = None, t: Union[float, list[float]] = None, return_intermediate: bool = False, complementary: bool = False, dense: bool = None, id=None, **odeint_kwargs, ): if not all(op.has_matrix for op in H.ops): raise ValueError( "All operators inside the parametrized hamiltonian must have a matrix defined." ) self._has_matrix = params is not None and t is not None self.H = H self.odeint_kwargs = odeint_kwargs if t is None: self.t = None else: if isinstance(t, (list, tuple)): t = qml.math.stack(t) self.t = qml.math.cast(qml.math.stack([0.0, t]) if qml.math.ndim(t) == 0 else t, float) if complementary and not return_intermediate: warnings.warn( "The keyword argument complementary does not have any effect if " "return_intermediate is set to False." ) if params is None: params = [] else: if not isinstance(H, HardwareHamiltonian) and len(params) != len(H.coeffs_parametrized): raise ValueError( "The length of the params argument and the number of scalar-valued functions " f"in the Hamiltonian must be the same. Received {len(params)=} parameters but " f"expected {len(H.coeffs_parametrized)} parameters." ) super().__init__(*params, wires=H.wires, id=id) self.hyperparameters["return_intermediate"] = return_intermediate self.hyperparameters["complementary"] = complementary self._check_time_batching() self.dense = len(self.wires) < 3 if dense is None else dense def __call__( self, params, t, return_intermediate=None, complementary=None, dense=None, **odeint_kwargs ): if not has_jax: raise ImportError( "Module jax is required for the ``ParametrizedEvolution`` class. " "You can install jax via: pip install jax" ) # Need to cast all elements inside params to `jnp.arrays` to make sure they are not cast # to `np.arrays` inside `Operator.__init__` params = [jnp.array(p) for p in params] # Inherit return_intermediate and complementary from self if not provided. if return_intermediate is None: return_intermediate = self.hyperparameters["return_intermediate"] if complementary is None: complementary = self.hyperparameters["complementary"] if dense is None: dense = self.dense odeint_kwargs = {**self.odeint_kwargs, **odeint_kwargs} if qml.QueuingManager.recording(): qml.QueuingManager.remove(self) return ParametrizedEvolution( H=self.H, params=params, t=t, return_intermediate=return_intermediate, complementary=complementary, dense=dense, id=self.id, **odeint_kwargs, ) def _check_time_batching(self): """Check whether the time argument is broadcasted/batched.""" if not self.hyperparameters["return_intermediate"] or self.t is None: return # Subtract 1 because the identity is never returned by `matrix`. If `complementary=True`, # subtract an additional 1 because the full time evolution is not being returned. self._batch_size = self.t.shape[0]
[docs] def map_wires(self, wire_map): mapped_op = super().map_wires(wire_map) mapped_op.H = self.H.map_wires(wire_map) return mapped_op
@property def hash(self): """int: Integer hash that uniquely represents the operator.""" return hash( ( str(self.name), tuple(self.wires.tolist()), str(self.hyperparameters.values()), str(self.t), str(self.data), self.H, str(self.odeint_kwargs.values()), ) ) def _flatten(self): data = self.data odeint_kwargs_tuples = tuple((key, value) for key, value in self.odeint_kwargs.items()) t = self.t if self.t is None else tuple(self.t) metadata = ( t, self.H, self.hyperparameters["return_intermediate"], self.hyperparameters["complementary"], self.dense, odeint_kwargs_tuples, ) return data, metadata @classmethod def _unflatten(cls, data, metadata): t, H, return_intermediate, complementary, dense, odeint_kwargs = metadata return cls( H, None if len(data) == 0 else data, t, return_intermediate=return_intermediate, complementary=complementary, dense=dense, **dict(odeint_kwargs), ) # pylint: disable=arguments-renamed, invalid-overridden-method @property def has_matrix(self): return self._has_matrix # pylint: disable=import-outside-toplevel
[docs] def matrix(self, wire_order=None): if not has_jax: raise ImportError( "Module jax is required for the ``ParametrizedEvolution`` class. " "You can install jax via: pip install jax" ) if not self.has_matrix: raise ValueError( "The parameters and the time window are required to compute the matrix. " "You can update its values by calling the class: EV(params, t)." ) y0 = jnp.eye(2 ** len(self.wires), dtype=complex) with jax.ensure_compile_time_eval(): H_jax = ParametrizedHamiltonianPytree.from_hamiltonian( self.H, dense=self.dense, wire_order=self.wires ) def fun(y, t): """dy/dt = -i H(t) y""" return (-1j * H_jax(self.data, t=t)) @ y mat = odeint(fun, y0, self.t, **self.odeint_kwargs) if self.hyperparameters["return_intermediate"] and self.hyperparameters["complementary"]: # Compute U(t_0, t_f)@U(t_0, t_i)^\dagger, where i indexes the first axis of mat mat = qml.math.tensordot(mat[-1], qml.math.conj(mat), axes=[[1], [-1]]) # The previous line leaves the axis indexing the t_i as second, so we move it up mat = qml.math.moveaxis(mat, 1, 0) elif not self.hyperparameters["return_intermediate"]: mat = mat[-1] return qml.math.expand_matrix(mat, wires=self.wires, wire_order=wire_order)
[docs] def label(self, decimals=None, base_label=None, cache=None): r"""A customizable string representation of the operator. Args: decimals=None (int): If ``None``, no parameters are included. Else, specifies how to round the parameters. base_label=None (str): overwrite the non-parameter component of the label cache=None (dict): dictionary that carries information between label calls in the same drawing Returns: str: label to use in drawings **Example:** >>> H = qml.X(1) + qml.pulse.constant * qml.Y(0) + jnp.polyval * qml.Y(1) >>> params = [0.2, [1, 2, 3]] >>> op = qml.evolve(H)(params, t=2) >>> cache = {'matrices': []} >>> op.label() "Parametrized\nEvolution" >>> op.label(decimals=2, cache=cache) "Parametrized\nEvolution\n(p=[0.20,M0], t=[0. 2.])" >>> op.label(base_label="my_label") "my_label" >>> op.label(decimals=2, base_label="my_label", cache=cache) "my_label\n(p=[0.20,M0], t=[0. 2.])" Array-like parameters are stored in ``cache['matrices']``. """ op_label = base_label or "Parametrized\nEvolution" if self.num_params == 0: return op_label if decimals is None: return op_label params = self.parameters has_cache = cache and isinstance(cache.get("matrices", None), list) if any(qml.math.ndim(p) for p in params) and not has_cache: return op_label def _format_number(x): return format(qml.math.toarray(x), f".{decimals}f") def _format_arraylike(x): for i, mat in enumerate(cache["matrices"]): if qml.math.shape(x) == qml.math.shape(mat) and qml.math.allclose(x, mat): return f"M{i}" mat_num = len(cache["matrices"]) cache["matrices"].append(x) return f"M{mat_num}" param_strings = [_format_arraylike(p) if p.shape else _format_number(p) for p in params] p = ",".join(s for s in param_strings) return f"{op_label}\n(p=[{p}], t={self.t})"
@functions.bind_new_parameters.register def _bind_new_parameters_parametrized_evol(op: ParametrizedEvolution, params: Sequence[TensorLike]): return ParametrizedEvolution( op.H, params=params, t=op.t, return_intermediate=op.hyperparameters["return_intermediate"], complementary=op.hyperparameters["complementary"], dense=op.dense, **op.odeint_kwargs, )