Skip to content

Commit

Permalink
Better way to determine if a matrix is a phase times a real matrix (#373
Browse files Browse the repository at this point in the history
)

* Add a better way to determine if a matrix is a phase times a real matrix and tests

* CHANGELOG updates

---------

Co-authored-by: Nicolas Quesada <nquesada@pop-os.localdomain>
  • Loading branch information
nquesada and Nicolas Quesada committed Aug 17, 2023
1 parent 8e43ac1 commit e999e49
Show file tree
Hide file tree
Showing 3 changed files with 27 additions and 2 deletions.
2 changes: 2 additions & 0 deletions .github/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@

* Simplifies the internal working of Williamson decomposition [(#366)](https://github.com/XanaduAI/thewalrus/pull/338).

* Improves the handling of an edge case in Takagi [(#373)](https://github.com/XanaduAI/thewalrus/pull/373).

### Bug fixes

### Documentation
Expand Down
6 changes: 4 additions & 2 deletions thewalrus/decompositions.py
Original file line number Diff line number Diff line change
Expand Up @@ -210,8 +210,10 @@ def takagi(A, svd_order=True):
list_vals.sort(reverse=svd_order)
sorted_ls, permutation = zip(*list_vals)
return np.array(sorted_ls), Uc[:, np.array(permutation)]

phi = np.angle(A[0, 0])
# Find the element with the largest absolute value
pos = np.unravel_index(np.argmax(np.abs(A)), (n, n))
# Use it to find whether the input is a global phase times a real matrix
phi = np.angle(A[pos])
Amr = np.real_if_close(np.exp(-1j * phi) * A)
if np.isrealobj(Amr):
vals, U = takagi(Amr, svd_order=svd_order)
Expand Down
21 changes: 21 additions & 0 deletions thewalrus/tests/test_decompositions.py
Original file line number Diff line number Diff line change
Expand Up @@ -394,3 +394,24 @@ def test_real_degenerate():
rl, U = takagi(mat)
assert np.allclose(U @ U.conj().T, np.eye(len(mat)))
assert np.allclose(U @ np.diag(rl) @ U.T, mat)


@pytest.mark.parametrize("size", [10, 20, 100])
def test_flat_phase(size):
"""Test that the correct decomposition is obtained even if the first entry is 0"""
A = np.random.rand(size, size) + 1j * np.random.rand(size, size)
A += A.T
A[0, 0] = 0
l, u = takagi(A)
assert np.allclose(A, u * l @ u.T)


def test_real_input_edge():
"""Adapted from https://math.stackexchange.com/questions/4418925/why-does-this-algorithm-for-the-takagi-factorization-fail-here"""
rng = np.random.default_rng(0) # Important for reproducibility
A = (rng.random((100, 100)) - 0.5) * 114
A = A * A.T # make A symmetric
l, u = takagi(A)
# Now, reconstruct A, see
Ar = u * l @ u.T
assert np.allclose(A, Ar)

0 comments on commit e999e49

Please sign in to comment.