Source code for toqito.channel_ops.complementary_channel
"""Computes the complementary channel/map of a superoperator."""
import numpy as np
[docs]
def complementary_channel(kraus_ops: list[np.ndarray]) -> list[np.ndarray]:
r"""Compute the Kraus operators for the complementary map of a quantum channel.
(Section: Representations and Characterizations of Channels from [@Watrous_2018_TQI]).
The complementary map is derived from the given quantum channel's Kraus operators by
rearranging the rows of the input Kraus operators into the Kraus operators of the
complementary map.
Specifically, for each Kraus operator \(K_i\) in the input channel \(\Phi\),
we define the complementary Kraus operators \(K_i^C\) by stacking the rows of
\(K_i\) from all Kraus operators vertically.
Examples:
Suppose the following Kraus operators define a quantum channel:
\[
K_1 = \frac{1}{\sqrt{2}} \begin{pmatrix}
1 & 0 \\
0 & 0
\end{pmatrix},
K_2 = \frac{1}{\sqrt{2}} \begin{pmatrix}
0 & 1 \\
0 & 0
\end{pmatrix},
K_3 = \frac{1}{\sqrt{2}} \begin{pmatrix}
0 & 0 \\
1 & 0
\end{pmatrix},
K_4 = \frac{1}{\sqrt{2}} \begin{pmatrix}
0 & 0 \\
0 & 1
\end{pmatrix}
\]
To compute the Kraus operators for the complementary map, we rearrange the rows of these
Kraus operators as follows:
```python exec="1" source="above"
import numpy as np
from toqito.channel_ops import complementary_channel
kraus_ops_Phi = [
np.sqrt(0.5) * np.array([[1, 0], [0, 0]]),
np.sqrt(0.5) * np.array([[0, 1], [0, 0]]),
np.sqrt(0.5) * np.array([[0, 0], [1, 0]]),
np.sqrt(0.5) * np.array([[0, 0], [0, 1]])
]
comp_kraus_ops = complementary_channel(kraus_ops_Phi)
for i, op in enumerate(comp_kraus_ops):
print(f"Kraus operator {i + 1}:")
print(op)
```
Raises:
ValueError: If the input is not a valid list of Kraus operators.
Args:
kraus_ops: A list of numpy arrays representing the Kraus operators of a quantum channel. Each Kraus operator is
assumed to be a square matrix.
Returns:
A list of numpy arrays representing the Kraus operators of the complementary map.
"""
num_kraus = len(kraus_ops)
if num_kraus == 0:
raise ValueError("All Kraus operators must be non-empty matrices.")
op_dim = kraus_ops[0].shape[0]
if any(k.shape[0] != k.shape[1] for k in kraus_ops):
raise ValueError("All Kraus operators must be square matrices.")
if any(k.shape[0] != op_dim for k in kraus_ops):
raise ValueError("All Kraus operators must be equal size matrices.")
# Check the Kraus completeness relation: ∑ K_i† K_i = I
identity = np.eye(op_dim, dtype=kraus_ops[0].dtype)
sum_k_dagger_k = sum(k.T.conj() @ k for k in kraus_ops)
if not np.allclose(sum_k_dagger_k, identity):
raise ValueError("The Kraus operators do not satisfy the completeness relation ∑ K_i† K_i = I.")
comp_kraus_ops = []
for row in range(op_dim):
comp_kraus_op = np.vstack([kraus_ops[i][row, :] for i in range(num_kraus)])
comp_kraus_ops.append(comp_kraus_op)
return comp_kraus_ops