Quantum information tools¶
This tutorial introduces the qinfo
module in QuAIRKit. The functions in quairkit.qinfo
can be categorized into several application areas relevant to quantum computation and quantum information. For quantum computation, there are functions mainly to compute the Kronecker product, conjugate transpose, and trace of matrices. There are also functions associated with quantum gates, such as decomposing a single-qubit unitary operation into rotation angles around the Z and Y axes. For functions
related to quantum information theory, this module includes partial trace, quantum entropies, the fidelity of quantum states, various kinds of norms, and so on. Additionally, this module provides data format validation such as verifying whether a matrix is a unitary, density matrix, or a projector.
Table of Contents - Functions in quantum computation - Functions in quantum information - Validation functions - Compatibility with different data formats
[1]:
import quairkit as qkit
from quairkit import to_state
from quairkit.database import *
from quairkit.qinfo import *
qkit.set_dtype("complex128")
Functions in quantum computation¶
First, initialize two random unitary matrices through a built-in function of QuAIRKit. The matrix A and B are both in torch.tensor format.
[2]:
A = random_unitary(num_qubits=1)
B = random_unitary(num_qubits=1)
print(f"Matrix A is:\n{A}\n")
print(f"Matrix B is:\n{B}")
Matrix A is:
tensor([[-0.6831+0.1926j, 0.1419+0.6900j],
[ 0.5660+0.4195j, -0.4625+0.5383j]])
Matrix B is:
tensor([[ 0.0309-0.3380j, 0.6674-0.6628j],
[ 0.9406+0.0033j, -0.2593-0.2190j]])
Users can use functions in qinfo
to implement specific operations on these two matrices. - Calculate the trace of a matrix by trace
. - Calculate the direct sum of matrix A and B with direct_sum
. Direct sum is an operation that combines two matrices into a larger matrix where A and B occupy diagonal blocks of the resulting matrix, and the remaining entries are zero. - Calculate the Kronecker products of at least two matrices by NKron
. - Implement conjugate transpose of a matrix by
dagger
. - Decompose a single-qubit unitary operator into Z-Y-Z rotation angles by decomp_1qubit
. - …
[3]:
print(f"The trace of matrix A is {trace(A)}")
print(f"The direct sum of matrix A and B is: \n{direct_sum(A,B)}\n")
print(f"The tensor product of matrix A and B is: \n{NKron(A,B)}\n")
print(f"The conjugate transpose of matrix A is: \n{dagger(A)}\n")
print(f"The decomposition of single-qubit unitary operator A to Z-Y-Z rotation angles is {decomp_1qubit(A)}")
The trace of matrix A is (-1.1456075082500372+0.7309499027665167j)
The direct sum of matrix A and B is:
tensor([[-0.6831+0.1926j, 0.1419+0.6900j, 0.0000+0.0000j, 0.0000+0.0000j],
[ 0.5660+0.4195j, -0.4625+0.5383j, 0.0000+0.0000j, 0.0000+0.0000j],
[ 0.0000+0.0000j, 0.0000+0.0000j, 0.0309-0.3380j, 0.6674-0.6628j],
[ 0.0000+0.0000j, 0.0000+0.0000j, 0.9406+0.0033j, -0.2593-0.2190j]])
The tensor product of matrix A and B is:
tensor([[ 0.0440+0.2368j, -0.3282+0.5813j, 0.2376-0.0267j, 0.5521+0.3665j],
[-0.6432+0.1789j, 0.2193+0.0996j, 0.1312+0.6495j, 0.1143-0.2100j],
[ 0.1593-0.1784j, 0.6558-0.0952j, 0.1677+0.1730j, 0.0481+0.6659j],
[ 0.5310+0.3964j, -0.0549-0.2327j, -0.4369+0.5048j, 0.2378-0.0383j]])
The conjugate transpose of matrix A is:
tensor([[-0.6831-0.1926j, 0.5660-0.4195j],
[ 0.1419-0.6900j, -0.4625-0.5383j]])
The decomposition of single-qubit unitary operator A to Z-Y-Z rotation angles is (tensor(-2.2289), tensor(1.5634), tensor(1.6428))
Functions in quantum information¶
Users can use functions in qinfo
to quantify information-theoretic properties of quantum states. - Calculate the von-Neumann entropy of a quantum state by von_neumann_entropy
. - Calculate the trace distance of two quantum states by trace_distance
. - Calculate the fidelity of two quantum states by state_fidelity
. - Calculate the purity of a quantum state by purity
. - Calculate the relative entropy of two quantum states by relative_entropy
. - Calculate of Schatten p-norm by
p_norm
. - …
[4]:
state1 = random_state(2).density_matrix
print(f"The first quantum state is:\n {state1}\n")
state2 = random_state(2, rank=1).ket
print(f"The second quantum state is:\n {state2}")
The first quantum state is:
tensor([[ 0.2031+0.0000j, 0.1826+0.0889j, -0.1991+0.0240j, 0.0808-0.2717j],
[ 0.1826-0.0889j, 0.2032+0.0000j, -0.1685+0.1088j, -0.0463-0.2798j],
[-0.1991-0.0240j, -0.1685-0.1088j, 0.1980+0.0000j, -0.1114+0.2568j],
[ 0.0808+0.2717j, -0.0463+0.2798j, -0.1114-0.2568j, 0.3958+0.0000j]])
The second quantum state is:
tensor([[ 0.6765+0.3616j],
[-0.4727-0.1993j],
[ 0.2708-0.0502j],
[-0.1490-0.2245j]])
[5]:
entropy = von_neumann_entropy(state1)
print(f"The von Neumann entropy between state 1 is:\n{entropy}")
traceDistance = trace_distance(state1, state2)
print(traceDistance.dtype)
print(f"The trace distance between state 1 and state 2 is:\n{traceDistance}")
fidelity = state_fidelity(state1, state2)
print(f"The state fidelity between state 1 and state 2 is:\n{fidelity}")
purity_state = purity(state1)
print(f"The purity of state 1 is:\n{purity_state}")
r_entropy = relative_entropy(state1, state2)
print(f"The relative entropy of state 1 and state 2 is:\n{r_entropy}")
p = 2
pnorm = p_norm(state1, p)
print(f"The Schatten {p}-norm of state 1 is:\n{pnorm}")
The von Neumann entropy between state 1 is:
-3.844111804577901e-15
torch.float64
The trace distance between state 1 and state 2 is:
0.989130913609007
The state fidelity between state 1 and state 2 is:
0.1470375336229326
The purity of state 1 is:
1.0000000000000009
The relative entropy of state 1 and state 2 is:
53.29869042772144
The Schatten 2-norm of state 1 is:
1.0000000000000004
Validation functions¶
Users can use validation functions in qinfo
to check if the matrix satisfies certain conditions. - Check if the input matrix is a positive semi-definite matrix by is_positive
. - Check if the input quantum state is PPT by is_ppt
. - Check if the input matrix is unitary by is_unitary
.
[6]:
is_positive(A)
print(f"The matrix is a positive matrix: {is_positive(A)}")
is_unitary(A)
print(f"The matrix is a unitary matrix: {is_unitary(A)}")
is_ppt(state1)
print(f"The state 1 is a PPT state: {is_ppt(state1)}")
The matrix is a positive matrix: False
The matrix is a unitary matrix: True
The state 1 is a PPT state: False
Compatibility with different data formats¶
Functions in qinfo
support different formats of input, including torch.Tensor, numpy.ndarray, and State. The input data format is transformed through _type_transform
. At the same time, the output format is consistent with the input format in the most situations. For example, the output will be numpy.ndarray format if the input is also numpy.ndarray format.
[7]:
# Input with torch.tensor format
print(f"State fidelity dtype (torch.tensor): {type(state_fidelity(state1, state2))}")
# Input with numpy.ndarray format
state1_num = state1.numpy()
state2_num = state2.numpy()
print(f"State fidelity dtype (numpy.ndarray): {type(state_fidelity(state1_num, state2_num))}")
# Input with State format
state1_sta = to_state(state1)
state2_sta = to_state(state2)
print(f"State fidelity dtype (State): {type(state_fidelity(state1_sta, state2_sta))}")
State fidelity dtype (torch.tensor): <class 'torch.Tensor'>
State fidelity dtype (numpy.ndarray): <class 'numpy.ndarray'>
State fidelity dtype (State): <class 'torch.Tensor'>
[8]:
qkit.print_info()
---------VERSION---------
quairkit: 0.3.0
torch: 2.5.1+cpu
numpy: 1.26.0
scipy: 1.14.1
matplotlib: 3.10.0
---------SYSTEM---------
Python version: 3.10.16
OS: Windows
OS version: 10.0.26100
---------DEVICE---------
CPU: ARMv8 (64-bit) Family 8 Model 1 Revision 201, Qualcomm Technologies Inc