Getting Started Guide

This guide provides information on the basic and more advanced usage of the AJD.jl package.

Installation

To install the Package follow these instructions to add the package to a basic Julia environment or use the package in a Pluto notebook.

Julia

Create temporary environment in the Julia REPL:

] activate --temp

Add the package:

] add https://github.com/muehlefeldt/AJD.jl

Pluto

The use in Pluto notebooks is supported. Add the package in one cell of the notebook:

begin
    using Pkg
    Pkg.activate("MyTestEnv")
    Pkg.add(url="https://github.com/muehlefeldt/AJD.jl")
    using AJD
end

Basic usage

The most basic usage of the package given a vector $M$ of $k$ square matrices ($n \times n$):

using AJD
filter = diagonalize(M)

The returned object filter of diagonalize() is a LinearFilter containing the matrix $F$ as well as $F'$. Assuming $A \in M$ these are used to diagonalize $A$ by calculating $D = F' A F$. $D$ being the diagonalized matrix. In Julia you calculate:

D = filter.iF * A * filter.F

Advanced options

The main function diagonalize() provides several options to be choosen by the user.

Algorithms

The package allows to choose different algorithms to calculate the AJD.

JDiag

Further reading on the ins and outs of the "JDiag" or "Jade" algorithm proposed be J.F. Cardoso can be found in reference [1].

Three implementations of the JDiag algorithm are available: JDiagCardoso(), JDiagGabrielDernbach() and JDiagEdourdPineau().

  • The Cardoso implementation is based on Matlab Code by Cardoso. Use the keyword algorithm = JDiagCardoso().
  • Based on a Python implementation by Gabrieldernbach the second implementation is suitable for matrices consisting of real and complex values. Use the keyword algorithm = JDiagGabrielDernbach().
  • The third implementation of JDiag is based on the Python code by Edouard Pineau also supports real and complex matrices, as well as hermitian matrices included in the module LinearAlgebra.jl, which are used in the Diagonalizations.jl package too. This implementation is currently the standard algorithm for the diagonalize() function. Use the keyword algorithm = JDiagEdourdPineau().

For example execute to execute :

using AJD

# Generate 1000 exactly diagonalizable matrices of 10 x 10 size.
M = AJD.get_test_data(:exact_diag, n_dims=10, n_matrices=1000)

# Diagonalize M using selected algorithm.
diagonalize(M, algorithm=JDiagEdourdPineau())

FFDiag

One implementation of the FFDiag algorithm is available through the keyword algorithm = FFDiag(). Resources on the topic can be found under reference [2].

If you want to use the FFDiag algorithm for calculation of the diagonalization be aware, that a set containing the same matrices, only one matrix or having the same elements on the offdiagonal symmetrical entries can't be calculated using the FFDiag algorithm. NaN values would occur when calculating the update matrix and the algorithm will return the identity matrix as a filter. If you want to calculate those sets refer to the JDiag implementations.

For a minimal example execute:

using AJD

# Generate 1000 exactly diagonalizable matrices of 10 x 10 size.
M = AJD.get_test_data(:exact_diag, n_dims=10, n_matrices=1000)

# Diagonalize M using selected algorithm.
diagonalize(M, algorithm=FFDiag())

Plotting

Visual feedback is optionally available. Plots of the convergence behaviour of the used algorithm, heatmaps of the filter matrix as well as the diagonalized matrices can be visualized using the symbol :plot. The diagonalized matrices $D_k$ are summarised and only $\overline{D_k}$ is plotted to provide an overview.

To generate a plot use diagonalize(M, :plot). Example code to generate a plot:

using AJD
using Plots

# Generate 200 exactly diagonalizable matrices of 50 x 50 size.
M = AJD.get_test_data(:exact_diag, n_dims=50, n_matrices=200)

# Diagonalize M and generate plot.
diagonalize(M, :plot, algorithm=FFDiag())

The generated plot:

Note

The plotting functionality is implemented as an extension to AJD.jl. This may require you to add the Plots.jl package to your Julia parent environment to use. This is done limit the dependency footprint of AJD.jl.

Benchmarking

An integrated benchmarking of the Jade and FFDiag algorithms is also available. Use diagonalize() with symbol :benchmark and specify the desired size matrices and number matrices. Large input sizes require a lot of computation time due to repeated execution. The algorithms are benchmarked using diffrent types of test data: exactly diagonalizable matrices, diagonalizable matrices and fully random matrices. Example code:

using AJD
using BenchmarkTools

# Using matrices of size 10 x 10. Use 1000 random matrices for each benchmarked run.
diagonalize(:benchmark, 10, 1000)

Note

The benchmarking functionality as well is implemented as an extension to AJD.jl. This may require you to add the BenchmarkTools package to your Julia parent environment. This is done limit the dependency footprint of AJD.jl as well.

Dependency on Diagonalizations.jl

The package works with the LinearFilter object of Diagonalizations.jl for further integration into the Diagonalizations.jl package if so desired. Furthermore, the LinearFilter gives rise to testing of the algorithms with functions like nonDiagonality provided as part of the package.