Represents a collection of complex values arranged in rows and columns.
Provides methods to operate algebraically on matrices.
Instantiation
ComplexMatrix objects can be created using different storage
schemes. To allocate storage for each matrix entry, so applying
a Dense scheme, one can exploit one of the
overloaded factory
method Dense, such as Dense(Int32, Int32).
On the contrary, method
Sparse returns
instances whose scheme
is StorageScheme.CompressedRow,
which means that storage is allocated for non-zero entries only, using
a compressed sparse row scheme.
Indexing
In a matrix, entries are arranged in rows and columns.
Zero-based indexes are assigned to rows and columns, so that
each entry can be identified by the indexes of the specific row
and column on which it lies.
Let
be a matrix, and consider its generic entry

where
and
are the
number of rows and columns of
, respectively.
Entry
can be set or get through the
indexer ItemInt32, Int32. Further overloads
of the indexer enable the access to sub-matrices, as well.
A linear index completely
identifies an entry,
assuming that entries are linearly ordered following the
ColumnMajor
data order. This means that entry
has linear
index equal to
, and matrix entries can be enumerated as
follows:

where
is the Count of the matrix.
Given a linear index, the corresponding entry can be accessed
via the indexer ItemInt32.
In order to retrieve entries corresponding to
multiple linear indexes simultaneously, linear indexers are also overloaded to accept
IndexCollection instances,
or strings as arguments.
Dimensions
The dimensions of a matrix can be inspected using the properties
NumberOfRows and
NumberOfColumns,
Use the property
Count
to know how many entries are arranged in a matrix.
The following table reports some particular dimensions
for which a property exists
which can be evaluated to inspect if the dimensions hold true for a given
matrix.
In addition, property IsSquare returns true
for matrices having the same number of rows and columns.
You can also inspect property IsVector to verify
if a matrix instance has only one row or column.
Diagonals
There are
diagonals in
.
The main diagonal of
is also said the diagonal
of order 0, and is the collection of
entries
such
that
.
If
, then the matrix has
super-diagonals: for
, the
-th super-diagonal is the collection of entries
corresponding to the positions

If
, the matrix has
sub-diagonals: for
, the
-th sub-diagonal is the collection of entries
corresponding to the positions

Bandwidths
The lower bandwidth of
is the smallest integer,
say
, such that
if
.
The upper bandwidth of a matrix
is the smallest integer,
say
, such that
if
.
These definitions imply that, if
, then the sub-diagonal
of order
contains all zero entries, and
if
, then the super-diagonal
of order
contains all zero entries.
Properties LowerBandwidth and UpperBandwidth can be
inspected to access the bandwidths of a given instance.
Patterns
The arrangement of zero entries in a matrix often follows a particular pattern.
Some properties are defined which can be tested to verify if
any of these patterns hold for a given matrix.
Relevant patterns can be described by means of lower and upper matrix
bandwidths.
For instance, an upper triangular matrix can be defined as a square matrix
having sub-diagonals whose entries are all zero, or, equivalently, as a
square matrix having a lower bandwidth equal to 0.
The following table reports some bandwidth-dependent patterns
for which a property exists
which can be evaluated to inspect if the pattern holds true for a given square
matrix.
Finally, property IsTriangular
can be inspected to verify if a matrix instance is lower or upper triangular,
IsBidiagonal to verify if
the instance is lower or upper bidiagonal,
while property IsHessenberg
returns a value indicating if a matrix instance is lower or upper Hessenberg.
Enumeration of entries and rows
Matrix values can be enumerated, and queried, using the "Linq to Objects" paradigm
(see, e.g., the
System.Linq
documentation). Matrices can also be enumerated by rows using the iterator
AsRowCollection.
This means that you can apply the LINQ approach to retrieve data from a matrix:
you write declarative code that describes what you want to
retrieve and at what conditions, which can be specified to
filter, order, and aggregate data as needed.
Memory usage
Matrix entries can be stored using different data schemes, as enumerated
in StorageScheme.
The data currently stored can be accessed as an array
through the
GetStorage method.
This method is intended for advanced users and must always be used
carefully.
For performance reasons, the returned reference points directly to
the matrix internal data. Do not call
GetStorage if
you do not have complete control of the instance you used to invoke
the method.
Method AsColumnMajorDenseArray always
returns a ColumnMajor ordered
array which is a dense representation of the matrix entries,
irrespective of the specific data scheme used to implement the matrix.
If the underlying scheme is
Dense, such method
returns a copy of the matrix data.
Serialization
Matrices can be loaded from, or saved to a CSV file through the
CsvComplexMatrixSerializer class.
Matrices can also be represented as JSON strings, see JsonSerialization.