Source code for toqito.perms.symmetric_projection
"""Symmetric projection operator produces a projection onto a symmetric subspace."""
import math
from itertools import permutations
import numpy as np
from scipy.linalg import orth
from toqito.perms import permutation_operator
[docs]
def symmetric_projection(dim: int, p_val: int = 2, partial: bool = False) -> np.ndarray:
r"""Produce the projection onto the symmetric subspace [@Chen_2014_Symmetric].
For a complex Euclidean space \(\mathcal{X}\) and a positive integer \(n\), the projection onto the
symmetric subspace is given by
\[
\frac{1}{n!} \sum_{\pi \in S_n} W_{\pi}
\]
where \(W_{\pi}\) is the swap operator and where \(S_n\) is the symmetric group on \(n\) symbols.
Produces the orthogonal projection onto the symmetric subspace of `p_val` copies of `dim`-dimensional space.
If `partial = True`, then the symmetric projection (PS) isn't the orthogonal projection itself, but rather a matrix
whose columns form an orthonormal basis for the symmetric subspace (and hence the PS * PS' is the orthogonal
projection onto the symmetric subspace).
This function was adapted from the QETLAB package.
Examples:
The \(2\)-dimensional symmetric projection with \(p=1\) is given as \(2\)-by-\(2\) identity matrix
\[
\begin{pmatrix}
1 & 0 \\
0 & 1
\end{pmatrix}.
\]
Using `|toqito⟩`, we can see this gives the proper result.
```python exec="1" source="above"
from toqito.perms import symmetric_projection
print(symmetric_projection(2, 1))
```
When \(d = 2\) and \(p = 2\) we have that
\[
\begin{pmatrix}
1 & 0 & 0 & 0 \\
0 & 1/2 & 1/2 & 0 \\
0 & 1/2 & 1/2 & 0 \\
0 & 0 & 0 & 1
\end{pmatrix}.
\]
Using `|toqito⟩` we can see this gives the proper result.
```python exec="1" source="above"
from toqito.perms import symmetric_projection
print(symmetric_projection(dim=2))
```
Args:
dim: The dimension of the local systems.
p_val: Default value of 2.
partial: Default value of 0.
Returns:
Projection onto the symmetric subspace.
"""
if dim < 1:
raise ValueError("InvalidDim: `dim` must be at least 1.")
if p_val < 1:
raise ValueError("InvalidPVal: `p_val` must be at least 1.")
dimp = dim**p_val
if p_val == 1:
return np.eye(dim)
p_list = np.array(list(permutations(np.arange(p_val))))
p_fac = math.factorial(p_val)
sym_proj = np.zeros((dimp, dimp))
for perm in p_list:
sym_proj += permutation_operator(dim * np.ones(p_val), perm, False, True)
sym_proj /= p_fac
if partial:
sym_proj = orth(sym_proj)
return sym_proj