Source code for toqito.state_props.concurrence
"""Concurrence property calculates the concurrence of a bipartite state.
The concurrence property is an entanglement measure defined for the product states of two qubits.
"""
import numpy as np
from toqito.matrices import pauli
[docs]
def concurrence(rho: np.ndarray) -> float:
r"""Calculate the concurrence of a bipartite state [@WikiConcurrence].
The concurrence of a bipartite state \(\rho\) is defined as
\[
\max(0, \lambda_1 - \lambda_2 - \lambda_3 - \lambda_4),
\]
where \(\lambda_1, \ldots, \lambda_4\) are the square roots of the
eigenvalues in decreasing order of the matrix
\[
\rho\tilde{\rho} = \rho \sigma_y \otimes \sigma_y \rho^* \sigma_y \otimes \sigma_y.
\]
Concurrence can serve as a measure of entanglement.
Examples:
Consider the following Bell state:
\[
u = \frac{1}{\sqrt{2}} \left( |00 \rangle + |11 \rangle \right).
\]
The concurrence of the density matrix \(\rho = u u^*\) defined by the vector \(u\) is
given as
\[
\mathcal{C}(\rho) \approx 1.
\]
The following example calculates this quantity using the `|toqito⟩` package.
```python exec="1" source="above"
import numpy as np
from toqito.matrices import standard_basis
from toqito.state_props import concurrence
e_0, e_1 = standard_basis(2)
e_00, e_11 = np.kron(e_0, e_0), np.kron(e_1, e_1)
u_vec = 1 / np.sqrt(2) * (e_00 + e_11)
rho = u_vec @ u_vec.conj().T
print(concurrence(rho))
```
Consider the concurrence of the following product state
\[
v = |0\rangle \otimes |1 \rangle.
\]
As this state has no entanglement, the concurrence is zero.
```python exec="1" source="above"
import numpy as np
from toqito.states import basis
from toqito.state_props import concurrence
e_0, e_1 = basis(2, 0), basis(2, 1)
v_vec = np.kron(e_0, e_1)
sigma = v_vec @ v_vec.conj().T
print(concurrence(sigma))
```
Raises:
ValueError: If system is not bipartite.
Args:
rho: The bipartite system specified as a matrix.
Returns:
The concurrence of the bipartite state \(\rho\).
"""
if rho.shape != (4, 4):
raise ValueError("InvalidDim: Concurrence is only defined for bipartite systems.")
sigma_y = pauli("Y", False)
sigma_y_y = np.kron(sigma_y, sigma_y)
rho_tilde = sigma_y_y @ rho.conj() @ sigma_y_y
eig_vals = np.linalg.eigvals(rho @ rho_tilde)
eig_vals = np.sort(np.abs(np.sqrt(eig_vals)))[::-1]
return max(0, eig_vals[0] - eig_vals[1] - eig_vals[2] - eig_vals[3])