Source code for toqito.channel_props.is_trace_preserving

"""Determines if a channel is trace-preserving."""

import numpy as np

from toqito.matrix_ops import partial_trace
from toqito.matrix_props import is_identity


[docs] def is_trace_preserving( phi: np.ndarray | list[list[np.ndarray]], rtol: float = 1e-05, atol: float = 1e-08, sys: int | list[int] = 2, dim: list[int] | np.ndarray | None = None, ) -> bool: r"""Determine whether the given channel is trace-preserving. A map \(\Phi \in \text{T} \left(\mathcal{X}, \mathcal{Y} \right)\) is *trace-preserving* if it holds that \[ \text{Tr} \left( \Phi(X) \right) = \text{Tr}\left( X \right) \] for every operator \(X \in \text{L}(\mathcal{X})\). Given the corresponding Choi matrix of the channel, a neccessary and sufficient condition is \[ \text{Tr}_{\mathcal{Y}} \left( J(\Phi) \right) = \mathbb{I}_{\mathcal{X}} \] In case `sys` is not specified, the default convention is that the Choi matrix is the result of applying the map to the second subsystem of the standard maximally entangled (unnormalized) state. The dimensions of the subsystems are given by the vector `dim`. By default, both subsystems have equal dimension. Alternatively, given a list of Kraus operators, a neccessary and sufficient condition is \[ \sum_{a \in \Sigma} A_a^* B_a = \mathbb{I}_{\mathcal{X}} \] Examples: The map \(\Phi\) defined as \[ \Phi(X) = X - U X U^* \] is not trace-preserving, where \[ U = \frac{1}{\sqrt{2}} \begin{pmatrix} 1 & 1 \\ -1 & 1 \end{pmatrix}. \] ```python exec="1" source="above" import numpy as np from toqito.channel_props import is_trace_preserving unitary_mat = np.array([[1, 1], [-1, 1]]) / np.sqrt(2) kraus_ops = [[np.identity(2), np.identity(2)], [unitary_mat, -unitary_mat]] print(is_trace_preserving(kraus_ops)) ``` As another example, the depolarizing channel is trace-preserving. ```python exec="1" source="above" from toqito.channels import depolarizing from toqito.channel_props import is_trace_preserving choi_mat = depolarizing(2) print(is_trace_preserving(choi_mat)) ``` Further information for determining the trace preserving properties of channels consult (Section: Linear Maps Of Square Operators from [@Watrous_2018_TQI]). Args: phi: The channel provided as either a Choi matrix or a list of Kraus operators. rtol: The relative tolerance parameter (default 1e-05). atol: The absolute tolerance parameter (default 1e-08). sys: Scalar or vector specifying the size of the subsystems. dim: Dimension of the subsystems. If `None`, all dimensions are assumed to be equal. Returns: True if the channel is trace-preserving, and False otherwise. """ # If the variable `phi` is provided as a list, we assume this is a list # of Kraus operators. if isinstance(phi, list): phi_l = [A for A, _ in phi] phi_r = [B for _, B in phi] k_l = np.concatenate(phi_l, axis=0) k_r = np.concatenate(phi_r, axis=0) mat = k_l.conj().T @ k_r else: if dim is None: n = phi.shape[0] d = int(round(np.sqrt(n))) if d * d != n: raise ValueError("Cannot infer equal subsystem dimensions. Please provide `dim`.") dim = [d, d] mat = partial_trace(phi, [sys - 1], dim) return is_identity(np.array(mat), rtol=rtol, atol=atol)