csharpfftfsharpintegrationinterpolationlinear-algebramathdifferentiationmatrixnumericsrandomregressionstatisticsmathnet
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1692 lines
68 KiB
1692 lines
68 KiB
// <copyright file="DiagonalMatrix.cs" company="Math.NET">
|
|
// Math.NET Numerics, part of the Math.NET Project
|
|
// http://numerics.mathdotnet.com
|
|
// http://github.com/mathnet/mathnet-numerics
|
|
// http://mathnetnumerics.codeplex.com
|
|
// Copyright (c) 2009-2010 Math.NET
|
|
// Permission is hereby granted, free of charge, to any person
|
|
// obtaining a copy of this software and associated documentation
|
|
// files (the "Software"), to deal in the Software without
|
|
// restriction, including without limitation the rights to use,
|
|
// copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following
|
|
// conditions:
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
// OTHER DEALINGS IN THE SOFTWARE.
|
|
// </copyright>
|
|
|
|
namespace MathNet.Numerics.LinearAlgebra.Complex
|
|
{
|
|
using System;
|
|
using System.Linq;
|
|
using System.Numerics;
|
|
using Distributions;
|
|
using Generic;
|
|
using Properties;
|
|
using Threading;
|
|
|
|
/// <summary>
|
|
/// A matrix type for diagonal matrices.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Diagonal matrices can be non-square matrices but the diagonal always starts
|
|
/// at element 0,0. A diagonal matrix will throw an exception if non diagonal
|
|
/// entries are set. The exception to this is when the off diagonal elements are
|
|
/// 0.0 or NaN; these settings will cause no change to the diagonal matrix.
|
|
/// </remarks>
|
|
public class DiagonalMatrix : Matrix<Complex>
|
|
{
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DiagonalMatrix"/> class. This matrix is square with a given size.
|
|
/// </summary>
|
|
/// <param name="order">the size of the square matrix.</param>
|
|
/// <exception cref="ArgumentException">
|
|
/// If <paramref name="order"/> is less than one.
|
|
/// </exception>
|
|
public DiagonalMatrix(int order) : base(order)
|
|
{
|
|
Data = new Complex[order * order];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DiagonalMatrix"/> class.
|
|
/// </summary>
|
|
/// <param name="rows">
|
|
/// The number of rows.
|
|
/// </param>
|
|
/// <param name="columns">
|
|
/// The number of columns.
|
|
/// </param>
|
|
public DiagonalMatrix(int rows, int columns) : base(rows, columns)
|
|
{
|
|
Data = new Complex[Math.Min(rows, columns)];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DiagonalMatrix"/> class with all entries set to a particular value.
|
|
/// </summary>
|
|
/// <param name="rows">
|
|
/// The number of rows.
|
|
/// </param>
|
|
/// <param name="columns">
|
|
/// The number of columns.
|
|
/// </param>
|
|
/// <param name="value">The value which we assign to each element of the matrix.</param>
|
|
public DiagonalMatrix(int rows, int columns, Complex value) : base(rows, columns)
|
|
{
|
|
Data = new Complex[Math.Min(rows, columns)];
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
Data[i] = value;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DiagonalMatrix"/> class from a one dimensional array with diagonal elements. This constructor
|
|
/// will reference the one dimensional array and not copy it.
|
|
/// </summary>
|
|
/// <param name="rows">The number of rows.</param>
|
|
/// <param name="columns">The number of columns.</param>
|
|
/// <param name="diagonalArray">The one dimensional array which contain diagonal elements.</param>
|
|
public DiagonalMatrix(int rows, int columns, Complex[] diagonalArray) : base(rows, columns)
|
|
{
|
|
Data = diagonalArray;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DiagonalMatrix"/> class from a 2D array.
|
|
/// </summary>
|
|
/// <param name="array">The 2D array to create this matrix from.</param>
|
|
/// <exception cref="IndexOutOfRangeException">When <paramref name="array"/> contains an off-diagonal element.</exception>
|
|
/// <exception cref="IndexOutOfRangeException">Depending on the implementation, an <see cref="IndexOutOfRangeException"/>
|
|
/// may be thrown if one of the indices is outside the dimensions of the matrix.</exception>
|
|
public DiagonalMatrix(Complex[,] array) : this(array.GetLength(0), array.GetLength(1))
|
|
{
|
|
var rows = array.GetLength(0);
|
|
var columns = array.GetLength(1);
|
|
|
|
for (var i = 0; i < rows; i++)
|
|
{
|
|
for (var j = 0; j < columns; j++)
|
|
{
|
|
if (i == j)
|
|
{
|
|
Data[i] = array[i, j];
|
|
}
|
|
else if (((array[i, j].Real != 0.0) && !double.IsNaN(array[i, j].Real)) || ((array[i, j].Imaginary != 0.0) && !double.IsNaN(array[i, j].Imaginary)))
|
|
{
|
|
throw new IndexOutOfRangeException("Cannot set an off-diagonal element in a diagonal matrix.");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the matrix's data.
|
|
/// </summary>
|
|
/// <value>The matrix's data.</value>
|
|
internal Complex[] Data
|
|
{
|
|
get;
|
|
private set;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieves the requested element without range checking.
|
|
/// </summary>
|
|
/// <param name="row">
|
|
/// The row of the element.
|
|
/// </param>
|
|
/// <param name="column">
|
|
/// The column of the element.
|
|
/// </param>
|
|
/// <returns>
|
|
/// The requested element.
|
|
/// </returns>
|
|
/// <exception cref="IndexOutOfRangeException">Depending on the implementation, an <see cref="IndexOutOfRangeException"/>
|
|
/// may be thrown if one of the indices is outside the dimensions of the matrix.</exception>
|
|
public override Complex At(int row, int column)
|
|
{
|
|
return row == column ? Data[row] : 0.0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the value of the given element.
|
|
/// </summary>
|
|
/// <param name="row">
|
|
/// The row of the element.
|
|
/// </param>
|
|
/// <param name="column">
|
|
/// The column of the element.
|
|
/// </param>
|
|
/// <param name="value">
|
|
/// The value to set the element to.
|
|
/// </param>
|
|
/// <exception cref="IndexOutOfRangeException">When trying to set an off diagonal element.</exception>
|
|
/// <exception cref="IndexOutOfRangeException">Depending on the implementation, an <see cref="IndexOutOfRangeException"/>
|
|
/// may be thrown if one of the indices is outside the dimensions of the matrix.</exception>
|
|
public override void At(int row, int column, Complex value)
|
|
{
|
|
if (row == column)
|
|
{
|
|
Data[row] = value;
|
|
}
|
|
else if (((value.Real != 0.0) && !double.IsNaN(value.Real)) || ((value.Imaginary != 0.0) && !double.IsNaN(value.Imaginary)))
|
|
{
|
|
throw new IndexOutOfRangeException("Cannot set an off-diagonal element in a diagonal matrix.");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a <c>DiagonalMatrix</c> for the given number of rows and columns.
|
|
/// </summary>
|
|
/// <param name="numberOfRows">
|
|
/// The number of rows.
|
|
/// </param>
|
|
/// <param name="numberOfColumns">
|
|
/// The number of columns.
|
|
/// </param>
|
|
/// <returns>
|
|
/// A <c>DiagonalMatrix</c> with the given dimensions.
|
|
/// </returns>
|
|
public override Matrix<Complex> CreateMatrix(int numberOfRows, int numberOfColumns)
|
|
{
|
|
return new DiagonalMatrix(numberOfRows, numberOfColumns);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a <see cref="Vector{T}"/> with a the given dimension.
|
|
/// </summary>
|
|
/// <param name="size">The size of the vector.</param>
|
|
/// <returns>
|
|
/// A <see cref="Vector{T}"/> with the given dimension.
|
|
/// </returns>
|
|
public override Vector<Complex> CreateVector(int size)
|
|
{
|
|
return new SparseVector(size);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets all values to zero.
|
|
/// </summary>
|
|
public override void Clear()
|
|
{
|
|
Array.Clear(Data, 0, Data.Length);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Indicates whether the current object is equal to another object of the same type.
|
|
/// </summary>
|
|
/// <param name="obj">
|
|
/// An object to compare with this object.
|
|
/// </param>
|
|
/// <returns>
|
|
/// <c>true</c> if the current object is equal to the <paramref name="obj"/> parameter; otherwise, <c>false</c>.
|
|
/// </returns>
|
|
public override bool Equals(object obj)
|
|
{
|
|
var diagonalMatrix = obj as DiagonalMatrix;
|
|
|
|
if (diagonalMatrix == null)
|
|
{
|
|
return base.Equals(obj);
|
|
}
|
|
|
|
// Accept if the argument is the same object as this
|
|
if (ReferenceEquals(this, diagonalMatrix))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (diagonalMatrix.Data.Length != Data.Length)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// If all else fails, perform element wise comparison.
|
|
return !Data.Where((t, i) => t != diagonalMatrix.Data[i]).Any();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a hash code for this instance.
|
|
/// </summary>
|
|
/// <returns>
|
|
/// A hash code for this instance, suitable for use in hashing algorithms and data structures like a hash table.
|
|
/// </returns>
|
|
public override int GetHashCode()
|
|
{
|
|
var hashNum = Math.Min(Data.Length, 25);
|
|
long hash = 0;
|
|
for (var i = 0; i < hashNum; i++)
|
|
{
|
|
#if SILVERLIGHT
|
|
hash ^= Precision.DoubleToInt64Bits(Data[i].GetHashCode());
|
|
#else
|
|
hash ^= BitConverter.DoubleToInt64Bits(Data[i].GetHashCode());
|
|
#endif
|
|
}
|
|
|
|
return BitConverter.ToInt32(BitConverter.GetBytes(hash), 4);
|
|
}
|
|
|
|
#region Elementary operations
|
|
/// <summary>
|
|
/// Adds another matrix to this matrix. The result will be written into this matrix.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to add to this matrix.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If the two matrices don't have the same dimensions.</exception>
|
|
/// <exception cref="ArgumentException">If <paramref name="other"/> is not <see cref="DiagonalMatrix"/>.</exception>
|
|
public override void Add(Matrix<Complex> other)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
var m = other as DiagonalMatrix;
|
|
if (m == null)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentTypeMismatch);
|
|
}
|
|
|
|
Add(m);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds another <see cref="DiagonalMatrix"/> to this matrix. The result will be written into this matrix.
|
|
/// </summary>
|
|
/// <param name="other">The <see cref="DiagonalMatrix"/> to add to this matrix.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If the two matrices don't have the same dimensions.</exception>
|
|
public void Add(DiagonalMatrix other)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
if (other.RowCount != RowCount || other.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
Control.LinearAlgebraProvider.AddArrays(Data, other.Data, Data);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Subtracts another matrix from this matrix. The result will be written into this matrix.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to subtract.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If the two matrices don't have the same dimensions.</exception>
|
|
/// <exception cref="ArgumentException">If <paramref name="other"/> is not <see cref="DiagonalMatrix"/>.</exception>
|
|
public override void Subtract(Matrix<Complex> other)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
var m = other as DiagonalMatrix;
|
|
if (m == null)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentTypeMismatch);
|
|
}
|
|
|
|
Subtract(m);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Subtracts another <see cref="DiagonalMatrix"/> from this matrix. The result will be written into this matrix.
|
|
/// </summary>
|
|
/// <param name="other">The <see cref="DiagonalMatrix"/> to subtract.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If the two matrices don't have the same dimensions.</exception>
|
|
public void Subtract(DiagonalMatrix other)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
if (other.RowCount != RowCount || other.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
Control.LinearAlgebraProvider.SubtractArrays(Data, other.Data, Data);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the values of the given array to the diagonal.
|
|
/// </summary>
|
|
/// <param name="source">The array to copy the values from. The length of the vector should be
|
|
/// Min(Rows, Columns).</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="source"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the length of <paramref name="source"/> does not
|
|
/// equal Min(Rows, Columns).</exception>
|
|
/// <remarks>For non-square matrices, the elements of <paramref name="source"/> are copied to
|
|
/// this[i,i].</remarks>
|
|
public override void SetDiagonal(Complex[] source)
|
|
{
|
|
if (source == null)
|
|
{
|
|
throw new ArgumentNullException("source");
|
|
}
|
|
|
|
if (source.Length != Data.Length)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentArraysSameLength, "source");
|
|
}
|
|
|
|
CommonParallel.For(0, source.Length, index => Data[index] = source[index]);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the values of the given <see cref="Vector{T}"/> to the diagonal.
|
|
/// </summary>
|
|
/// <param name="source">The vector to copy the values from. The length of the vector should be
|
|
/// Min(Rows, Columns).</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="source"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the length of <paramref name="source"/> does not
|
|
/// equal Min(Rows, Columns).</exception>
|
|
/// <remarks>For non-square matrices, the elements of <paramref name="source"/> are copied to
|
|
/// this[i,i].</remarks>
|
|
public override void SetDiagonal(Vector<Complex> source)
|
|
{
|
|
var denseSource = source as DenseVector;
|
|
if (denseSource == null)
|
|
{
|
|
base.SetDiagonal(source);
|
|
return;
|
|
}
|
|
|
|
if (Data.Length != denseSource.Data.Length)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentVectorsSameLength, "source");
|
|
}
|
|
|
|
CommonParallel.For(0, denseSource.Data.Length, index => Data[index] = denseSource.Data[index]);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies each element of this matrix with a scalar.
|
|
/// </summary>
|
|
/// <param name="scalar">The scalar to multiply with.</param>
|
|
public override void Multiply(Complex scalar)
|
|
{
|
|
if (scalar == 0.0)
|
|
{
|
|
Clear();
|
|
return;
|
|
}
|
|
|
|
if (scalar == 1.0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
Control.LinearAlgebraProvider.ScaleArray(scalar, Data);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies this diagonal matrix with another diagonal matrix and places the results into the result diagonal matrix.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to multiply with.</param>
|
|
/// <param name="result">The result of the multiplication.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If the result matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>this.Columns != other.Rows</strong>.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the this.Rows x other.Columns.</exception>
|
|
public override void Multiply(Matrix<Complex> other, Matrix<Complex> result)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (ColumnCount != other.RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
if (result.RowCount != RowCount || result.ColumnCount != other.ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
var m = other as DiagonalMatrix;
|
|
var r = result as DiagonalMatrix;
|
|
|
|
if (m == null || r == null)
|
|
{
|
|
base.Multiply(other, result);
|
|
}
|
|
else
|
|
{
|
|
var thisDataCopy = new Complex[r.Data.Length];
|
|
var otherDataCopy = new Complex[r.Data.Length];
|
|
|
|
CommonParallel.For(0, (r.Data.Length > Data.Length) ? Data.Length : r.Data.Length, index => thisDataCopy[index] = Data[index]);
|
|
CommonParallel.For(0, (r.Data.Length > m.Data.Length) ? m.Data.Length : r.Data.Length, index => otherDataCopy[index] = m.Data[index]);
|
|
|
|
Control.LinearAlgebraProvider.PointWiseMultiplyArrays(thisDataCopy, otherDataCopy, r.Data);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies this matrix with another matrix and returns the result.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to multiply with.</param>
|
|
/// <exception cref="ArgumentException">If <strong>this.Columns != other.Rows</strong>.</exception>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <returns>The result of multiplication.</returns>
|
|
public override Matrix<Complex> Multiply(Matrix<Complex> other)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
if (ColumnCount != other.RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
var m = other as DiagonalMatrix;
|
|
if (m == null)
|
|
{
|
|
return base.Multiply(other);
|
|
}
|
|
|
|
var result = (DiagonalMatrix)CreateMatrix(RowCount, other.ColumnCount);
|
|
Multiply(other, result);
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies this matrix with a vector and places the results into the result matrix.
|
|
/// </summary>
|
|
/// <param name="rightSide">The vector to multiply with.</param>
|
|
/// <param name="result">The result of the multiplication.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="rightSide"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="result"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>result.Count != this.RowCount</strong>.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>this.ColumnCount != <paramref name="rightSide"/>.Count</strong>.</exception>
|
|
public override void Multiply(Vector<Complex> rightSide, Vector<Complex> result)
|
|
{
|
|
if (rightSide == null)
|
|
{
|
|
throw new ArgumentNullException("rightSide");
|
|
}
|
|
|
|
if (ColumnCount != rightSide.Count)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "rightSide");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (RowCount != result.Count)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
if (ReferenceEquals(rightSide, result))
|
|
{
|
|
var tmp = result.CreateVector(result.Count);
|
|
Multiply(rightSide, tmp);
|
|
tmp.CopyTo(result);
|
|
}
|
|
else
|
|
{
|
|
// Clear the result vector
|
|
result.Clear();
|
|
|
|
// Multiply the elements in the vector with the corresponding diagonal element in this.
|
|
for (var r = 0; r < Data.Length; r++)
|
|
{
|
|
result[r] = Data[r] * rightSide[r];
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Left multiply a matrix with a vector ( = vector * matrix ) and place the result in the result vector.
|
|
/// </summary>
|
|
/// <param name="leftSide">The vector to multiply with.</param>
|
|
/// <param name="result">The result of the multiplication.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="leftSide"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If the result matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>result.Count != this.ColumnCount</strong>.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>this.RowCount != <paramref name="leftSide"/>.Count</strong>.</exception>
|
|
public override void LeftMultiply(Vector<Complex> leftSide, Vector<Complex> result)
|
|
{
|
|
if (leftSide == null)
|
|
{
|
|
throw new ArgumentNullException("leftSide");
|
|
}
|
|
|
|
if (RowCount != leftSide.Count)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "leftSide");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (ColumnCount != result.Count)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
if (ReferenceEquals(leftSide, result))
|
|
{
|
|
var tmp = result.CreateVector(result.Count);
|
|
LeftMultiply(leftSide, tmp);
|
|
tmp.CopyTo(result);
|
|
}
|
|
else
|
|
{
|
|
// Clear the result vector
|
|
result.Clear();
|
|
|
|
// Multiply the elements in the vector with the corresponding diagonal element in this.
|
|
for (var r = 0; r < Data.Length; r++)
|
|
{
|
|
result[r] = Data[r] * leftSide[r];
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Computes the determinant of this matrix.
|
|
/// </summary>
|
|
/// <returns>The determinant of this matrix.</returns>
|
|
public override Complex Determinant()
|
|
{
|
|
if (RowCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSquare);
|
|
}
|
|
|
|
return Data.Aggregate(Complex.One, (current, t) => current * t);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the elements of the diagonal in a <see cref="DenseVector"/>.
|
|
/// </summary>
|
|
/// <returns>The elements of the diagonal.</returns>
|
|
/// <remarks>For non-square matrices, the method returns Min(Rows, Columns) elements where
|
|
/// i == j (i is the row index, and j is the column index).</remarks>
|
|
public override Vector<Complex> Diagonal()
|
|
{
|
|
// TODO: Should we return reference to array? In current implementation we return copy of array, so changes in DenseVector will
|
|
// not influence onto diagonal elements
|
|
return new DenseVector((Complex[])Data.Clone());
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies this diagonal matrix with transpose of another diagonal matrix and places the results into the result diagonal matrix.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to multiply with.</param>
|
|
/// <param name="result">The result of the multiplication.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If the result matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>this.Columns != other.Rows</strong>.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the this.Rows x other.Columns.</exception>
|
|
public override void TransposeAndMultiply(Matrix<Complex> other, Matrix<Complex> result)
|
|
{
|
|
var otherDiagonal = other as DiagonalMatrix;
|
|
var resultDiagonal = result as DiagonalMatrix;
|
|
|
|
if (otherDiagonal == null || resultDiagonal == null)
|
|
{
|
|
base.TransposeAndMultiply(other, result);
|
|
return;
|
|
}
|
|
|
|
Multiply(otherDiagonal.Transpose(), result);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies this matrix with transpose of another matrix and returns the result.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to multiply with.</param>
|
|
/// <exception cref="ArgumentException">If <strong>this.Columns != other.Rows</strong>.</exception>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <returns>The result of multiplication.</returns>
|
|
public override Matrix<Complex> TransposeAndMultiply(Matrix<Complex> other)
|
|
{
|
|
var otherDiagonal = other as DiagonalMatrix;
|
|
if (otherDiagonal == null)
|
|
{
|
|
return base.TransposeAndMultiply(other);
|
|
}
|
|
|
|
if (ColumnCount != otherDiagonal.ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
var result = (DiagonalMatrix)CreateMatrix(RowCount, other.RowCount);
|
|
TransposeAndMultiply(other, result);
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiplies two diagonal matrices.
|
|
/// </summary>
|
|
/// <param name="leftSide">The left matrix to multiply.</param>
|
|
/// <param name="rightSide">The right matrix to multiply.</param>
|
|
/// <returns>The result of multiplication.</returns>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="leftSide"/> or <paramref name="rightSide"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the dimensions of <paramref name="leftSide"/> or <paramref name="rightSide"/> don't conform.</exception>
|
|
public static DiagonalMatrix operator *(DiagonalMatrix leftSide, DiagonalMatrix rightSide)
|
|
{
|
|
if (leftSide == null)
|
|
{
|
|
throw new ArgumentNullException("leftSide");
|
|
}
|
|
|
|
if (rightSide == null)
|
|
{
|
|
throw new ArgumentNullException("rightSide");
|
|
}
|
|
|
|
if (leftSide.ColumnCount != rightSide.RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions);
|
|
}
|
|
|
|
return (DiagonalMatrix)leftSide.Multiply(rightSide);
|
|
}
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Copies the elements of this matrix to the given matrix.
|
|
/// </summary>
|
|
/// <param name="target">
|
|
/// The matrix to copy values into.
|
|
/// </param>
|
|
/// <exception cref="ArgumentNullException">
|
|
/// If target is <see langword="null"/>.
|
|
/// </exception>
|
|
/// <exception cref="ArgumentException">
|
|
/// If this and the target matrix do not have the same dimensions..
|
|
/// </exception>
|
|
public override void CopyTo(Matrix<Complex> target)
|
|
{
|
|
var diagonalTarget = target as DiagonalMatrix;
|
|
|
|
if (diagonalTarget == null)
|
|
{
|
|
base.CopyTo(target);
|
|
return;
|
|
}
|
|
|
|
if (ReferenceEquals(this, target))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (RowCount != target.RowCount || ColumnCount != target.ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "target");
|
|
}
|
|
|
|
CommonParallel.For(0, Data.Length, index => diagonalTarget.Data[index] = Data[index]);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the transpose of this matrix.
|
|
/// </summary>
|
|
/// <returns>The transpose of this matrix.</returns>
|
|
public override Matrix<Complex> Transpose()
|
|
{
|
|
var ret = new DiagonalMatrix(ColumnCount, RowCount);
|
|
CommonParallel.For(0, Data.Length, index => ret.Data[index] = Data[index]);
|
|
return ret;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the conjugate transpose of this matrix.
|
|
/// </summary>
|
|
/// <returns>The conjugate transpose of this matrix.</returns>
|
|
public override Matrix<Complex> ConjugateTranspose()
|
|
{
|
|
var ret = new DiagonalMatrix(ColumnCount, RowCount);
|
|
CommonParallel.For(0, Data.Length, index => ret.Data[index] = Data[index].Conjugate());
|
|
return ret;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the requested column elements into the given vector.
|
|
/// </summary>
|
|
/// <param name="columnIndex">The column to copy elements from.</param>
|
|
/// <param name="rowIndex">The row to start copying from.</param>
|
|
/// <param name="length">The number of elements to copy.</param>
|
|
/// <param name="result">The <see cref="Vector{T}"/> to copy the column into.</param>
|
|
/// <exception cref="ArgumentNullException">If the result <see cref="Vector{T}"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="columnIndex"/> is negative,
|
|
/// or greater than or equal to the number of columns.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="rowIndex"/> is negative,
|
|
/// or greater than or equal to the number of rows.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="rowIndex"/> + <paramref name="length"/>
|
|
/// is greater than or equal to the number of rows.</exception>
|
|
/// <exception cref="ArgumentException">If <paramref name="length"/> is not positive.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <strong>result.Count < length</strong>.</exception>
|
|
public override void Column(int columnIndex, int rowIndex, int length, Vector<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (columnIndex >= ColumnCount || columnIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("columnIndex");
|
|
}
|
|
|
|
if (rowIndex >= RowCount || rowIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("rowIndex");
|
|
}
|
|
|
|
if (rowIndex + length > RowCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("length");
|
|
}
|
|
|
|
if (length < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "length");
|
|
}
|
|
|
|
if (result.Count < length)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentVectorsSameLength, "result");
|
|
}
|
|
|
|
// Clear the result and copy the diagonal entry.
|
|
result.Clear();
|
|
if (columnIndex >= rowIndex && columnIndex < rowIndex + length && columnIndex < Data.Length)
|
|
{
|
|
result[columnIndex - rowIndex] = Data[columnIndex];
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Copies the requested row elements into a new <see cref="Vector{T}"/>.
|
|
/// </summary>
|
|
/// <param name="rowIndex">The row to copy elements from.</param>
|
|
/// <param name="columnIndex">The column to start copying from.</param>
|
|
/// <param name="length">The number of elements to copy.</param>
|
|
/// <param name="result">The <see cref="Vector{T}"/> to copy the column into.</param>
|
|
/// <exception cref="ArgumentNullException">If the result <see cref="Vector{T}"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="rowIndex"/> is negative,
|
|
/// or greater than or equal to the number of columns.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="columnIndex"/> is negative,
|
|
/// or greater than or equal to the number of rows.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="columnIndex"/> + <paramref name="length"/>
|
|
/// is greater than or equal to the number of rows.</exception>
|
|
/// <exception cref="ArgumentException">If <paramref name="length"/> is not positive.</exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <strong>result.Count < length</strong>.</exception>
|
|
public override void Row(int rowIndex, int columnIndex, int length, Vector<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (rowIndex >= RowCount || rowIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("rowIndex");
|
|
}
|
|
|
|
if (columnIndex >= ColumnCount || columnIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("columnIndex");
|
|
}
|
|
|
|
if (columnIndex + length > ColumnCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("length");
|
|
}
|
|
|
|
if (length < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "length");
|
|
}
|
|
|
|
if (result.Count < length)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentVectorsSameLength, "result");
|
|
}
|
|
|
|
// Clear the result and copy the diagonal entry.
|
|
result.Clear();
|
|
if (rowIndex >= columnIndex && rowIndex < columnIndex + length && rowIndex < Data.Length)
|
|
{
|
|
result[rowIndex - columnIndex] = Data[rowIndex];
|
|
}
|
|
}
|
|
|
|
/// <summary>Calculates the L1 norm.</summary>
|
|
/// <returns>The L1 norm of the matrix.</returns>
|
|
public override double L1Norm()
|
|
{
|
|
return Data.Aggregate(double.NegativeInfinity, (current, t) => Math.Max(current, t.Magnitude));
|
|
}
|
|
|
|
/// <summary>Calculates the L2 norm.</summary>
|
|
/// <returns>The L2 norm of the matrix.</returns>
|
|
public override double L2Norm()
|
|
{
|
|
return Data.Aggregate(double.NegativeInfinity, (current, t) => Math.Max(current, t.Magnitude));
|
|
}
|
|
|
|
/// <summary>Calculates the Frobenius norm of this matrix.</summary>
|
|
/// <returns>The Frobenius norm of this matrix.</returns>
|
|
public override double FrobeniusNorm()
|
|
{
|
|
var norm = Data.Sum(t => t.Magnitude * t.Magnitude);
|
|
return Math.Sqrt(norm);
|
|
}
|
|
|
|
/// <summary>Calculates the infinity norm of this matrix.</summary>
|
|
/// <returns>The infinity norm of this matrix.</returns>
|
|
public override double InfinityNorm()
|
|
{
|
|
return L1Norm();
|
|
}
|
|
|
|
/// <summary>Calculates the condition number of this matrix.</summary>
|
|
/// <returns>The condition number of the matrix.</returns>
|
|
public override double ConditionNumber()
|
|
{
|
|
var maxSv = double.NegativeInfinity;
|
|
var minSv = double.PositiveInfinity;
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
maxSv = Math.Max(maxSv, Data[i].Magnitude);
|
|
minSv = Math.Min(minSv, Data[i].Magnitude);
|
|
}
|
|
|
|
return maxSv / minSv;
|
|
}
|
|
|
|
/// <summary>Computes the inverse of this matrix.</summary>
|
|
/// <exception cref="ArgumentException">If <see cref="DiagonalMatrix"/> is not a square matrix.</exception>
|
|
/// <exception cref="ArgumentException">If <see cref="DiagonalMatrix"/> is singular.</exception>
|
|
/// <returns>The inverse of this matrix.</returns>
|
|
public override Matrix<Complex> Inverse()
|
|
{
|
|
if (RowCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSquare);
|
|
}
|
|
|
|
var inverse = (DiagonalMatrix)Clone();
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
if (Data[i] != 0.0)
|
|
{
|
|
inverse.Data[i] = 1.0 / Data[i];
|
|
}
|
|
else
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixNotSingular);
|
|
}
|
|
}
|
|
|
|
return inverse;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a new matrix containing the lower triangle of this matrix.
|
|
/// </summary>
|
|
/// <returns>The lower triangle of this matrix.</returns>
|
|
public override Matrix<Complex> LowerTriangle()
|
|
{
|
|
return Clone();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Puts the lower triangle of this matrix into the result matrix.
|
|
/// </summary>
|
|
/// <param name="result">Where to store the lower triangle.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="result"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the same as this matrix.</exception>
|
|
public override void LowerTriangle(Matrix<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != RowCount || result.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
if (ReferenceEquals(this, result))
|
|
{
|
|
return;
|
|
}
|
|
|
|
result.Clear();
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a new matrix containing the lower triangle of this matrix. The new matrix
|
|
/// does not contain the diagonal elements of this matrix.
|
|
/// </summary>
|
|
/// <returns>The lower triangle of this matrix.</returns>
|
|
public override Matrix<Complex> StrictlyLowerTriangle()
|
|
{
|
|
return new DiagonalMatrix(RowCount, ColumnCount);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Puts the strictly lower triangle of this matrix into the result matrix.
|
|
/// </summary>
|
|
/// <param name="result">Where to store the lower triangle.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="result"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the same as this matrix.</exception>
|
|
public override void StrictlyLowerTriangle(Matrix<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != RowCount || result.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
result.Clear();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a new matrix containing the upper triangle of this matrix.
|
|
/// </summary>
|
|
/// <returns>The upper triangle of this matrix.</returns>
|
|
public override Matrix<Complex> UpperTriangle()
|
|
{
|
|
return Clone();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Puts the upper triangle of this matrix into the result matrix.
|
|
/// </summary>
|
|
/// <param name="result">Where to store the lower triangle.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="result"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the same as this matrix.</exception>
|
|
public override void UpperTriangle(Matrix<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != RowCount || result.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
result.Clear();
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a new matrix containing the upper triangle of this matrix. The new matrix
|
|
/// does not contain the diagonal elements of this matrix.
|
|
/// </summary>
|
|
/// <returns>The upper triangle of this matrix.</returns>
|
|
public override Matrix<Complex> StrictlyUpperTriangle()
|
|
{
|
|
return new DiagonalMatrix(RowCount, ColumnCount);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Puts the strictly upper triangle of this matrix into the result matrix.
|
|
/// </summary>
|
|
/// <param name="result">Where to store the lower triangle.</param>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="result"/> is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not the same as this matrix.</exception>
|
|
public override void StrictlyUpperTriangle(Matrix<Complex> result)
|
|
{
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != RowCount || result.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
result.Clear();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a matrix that contains the values from the requested sub-matrix.
|
|
/// </summary>
|
|
/// <param name="rowIndex">The row to start copying from.</param>
|
|
/// <param name="rowLength">The number of rows to copy. Must be positive.</param>
|
|
/// <param name="columnIndex">The column to start copying from.</param>
|
|
/// <param name="columnLength">The number of columns to copy. Must be positive.</param>
|
|
/// <returns>The requested sub-matrix.</returns>
|
|
/// <exception cref="ArgumentOutOfRangeException">If: <list><item><paramref name="rowIndex"/> is
|
|
/// negative, or greater than or equal to the number of rows.</item>
|
|
/// <item><paramref name="columnIndex"/> is negative, or greater than or equal to the number
|
|
/// of columns.</item>
|
|
/// <item><c>(columnIndex + columnLength) >= Columns</c></item>
|
|
/// <item><c>(rowIndex + rowLength) >= Rows</c></item></list></exception>
|
|
/// <exception cref="ArgumentException">If <paramref name="rowLength"/> or <paramref name="columnLength"/>
|
|
/// is not positive.</exception>
|
|
public override Matrix<Complex> SubMatrix(int rowIndex, int rowLength, int columnIndex, int columnLength)
|
|
{
|
|
if (rowIndex >= RowCount || rowIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("rowIndex");
|
|
}
|
|
|
|
if (columnIndex >= ColumnCount || columnIndex < 0)
|
|
{
|
|
throw new ArgumentOutOfRangeException("columnIndex");
|
|
}
|
|
|
|
if (rowLength < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "rowLength");
|
|
}
|
|
|
|
if (columnLength < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "columnLength");
|
|
}
|
|
|
|
var colMax = columnIndex + columnLength;
|
|
var rowMax = rowIndex + rowLength;
|
|
|
|
if (rowMax > RowCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("rowLength");
|
|
}
|
|
|
|
if (colMax > ColumnCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("columnLength");
|
|
}
|
|
|
|
var result = new SparseMatrix(rowLength, columnLength);
|
|
|
|
if (rowIndex > columnIndex && columnIndex + columnLength > rowIndex)
|
|
{
|
|
for (var i = 0; rowIndex - columnIndex + i < Math.Min(columnLength, rowLength); i++)
|
|
{
|
|
result[i, rowIndex - columnIndex + i] = Data[rowIndex + i];
|
|
}
|
|
}
|
|
else if (rowIndex < columnIndex && rowIndex + rowLength > columnIndex)
|
|
{
|
|
for (var i = 0; rowIndex - columnIndex + i < Math.Min(columnLength, rowLength); i++)
|
|
{
|
|
result[columnIndex - rowIndex + i, i] = Data[columnIndex + i];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (var i = 0; i < Math.Min(columnLength, rowLength); i++)
|
|
{
|
|
result[i, i] = Data[rowIndex + i];
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns this matrix as a multidimensional array.
|
|
/// </summary>
|
|
/// <returns>A multidimensional containing the values of this matrix.</returns>
|
|
public override Complex[,] ToArray()
|
|
{
|
|
var result = new Complex[RowCount, ColumnCount];
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="SparseMatrix"/> and inserts the given column at the given index.
|
|
/// </summary>
|
|
/// <param name="columnIndex">The index of where to insert the column.</param>
|
|
/// <param name="column">The column to insert.</param>
|
|
/// <returns>A new <see cref="SparseMatrix"/> with the inserted column.</returns>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="column "/> is <see langword="null" />. </exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="columnIndex"/> is < zero or > the number of columns.</exception>
|
|
/// <exception cref="ArgumentException">If the size of <paramref name="column"/> != the number of rows.</exception>
|
|
public override Matrix<Complex> InsertColumn(int columnIndex, Vector<Complex> column)
|
|
{
|
|
if (column == null)
|
|
{
|
|
throw new ArgumentNullException("column");
|
|
}
|
|
|
|
if (columnIndex < 0 || columnIndex > ColumnCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("columnIndex");
|
|
}
|
|
|
|
if (column.Count != RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameRowDimension, "column");
|
|
}
|
|
|
|
var result = new SparseMatrix(RowCount, ColumnCount + 1);
|
|
|
|
for (var i = 0; i < columnIndex; i++)
|
|
{
|
|
result.SetColumn(i, Column(i));
|
|
}
|
|
|
|
result.SetColumn(columnIndex, column);
|
|
|
|
for (var i = columnIndex + 1; i < ColumnCount + 1; i++)
|
|
{
|
|
result.SetColumn(i, Column(i - 1));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new <see cref="SparseMatrix"/> and inserts the given row at the given index.
|
|
/// </summary>
|
|
/// <param name="rowIndex">The index of where to insert the row.</param>
|
|
/// <param name="row">The row to insert.</param>
|
|
/// <returns>A new <see cref="SparseMatrix"/> with the inserted column.</returns>
|
|
/// <exception cref="ArgumentNullException">If <paramref name="row"/> is <see langword="null" />. </exception>
|
|
/// <exception cref="ArgumentOutOfRangeException">If <paramref name="rowIndex"/> is < zero or > the number of rows.</exception>
|
|
/// <exception cref="ArgumentException">If the size of <paramref name="row"/> != the number of columns.</exception>
|
|
public override Matrix<Complex> InsertRow(int rowIndex, Vector<Complex> row)
|
|
{
|
|
if (row == null)
|
|
{
|
|
throw new ArgumentNullException("row");
|
|
}
|
|
|
|
if (rowIndex < 0 || rowIndex > RowCount)
|
|
{
|
|
throw new ArgumentOutOfRangeException("rowIndex");
|
|
}
|
|
|
|
if (row.Count != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameRowDimension, "row");
|
|
}
|
|
|
|
var result = new SparseMatrix(RowCount + 1, ColumnCount);
|
|
|
|
for (var i = 0; i < rowIndex; i++)
|
|
{
|
|
result.SetRow(i, Row(i));
|
|
}
|
|
|
|
result.SetRow(rowIndex, row);
|
|
|
|
for (var i = rowIndex + 1; i < RowCount; i++)
|
|
{
|
|
result.SetRow(i, Row(i - 1));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stacks this matrix on top of the given matrix and places the result into the result <see cref="SparseMatrix"/>.
|
|
/// </summary>
|
|
/// <param name="lower">The matrix to stack this matrix upon.</param>
|
|
/// <returns>The combined <see cref="SparseMatrix"/>.</returns>
|
|
/// <exception cref="ArgumentNullException">If lower is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>upper.Columns != lower.Columns</strong>.</exception>
|
|
public override Matrix<Complex> Stack(Matrix<Complex> lower)
|
|
{
|
|
if (lower == null)
|
|
{
|
|
throw new ArgumentNullException("lower");
|
|
}
|
|
|
|
if (lower.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameColumnDimension, "lower");
|
|
}
|
|
|
|
var result = new SparseMatrix(RowCount + lower.RowCount, ColumnCount);
|
|
Stack(lower, result);
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stacks this matrix on top of the given matrix and places the result into the result <see cref="SparseMatrix"/>.
|
|
/// </summary>
|
|
/// <param name="lower">The matrix to stack this matrix upon.</param>
|
|
/// <param name="result">The combined <see cref="SparseMatrix"/>.</param>
|
|
/// <exception cref="ArgumentNullException">If lower is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If <strong>upper.Columns != lower.Columns</strong>.</exception>
|
|
public override void Stack(Matrix<Complex> lower, Matrix<Complex> result)
|
|
{
|
|
if (lower == null)
|
|
{
|
|
throw new ArgumentNullException("lower");
|
|
}
|
|
|
|
if (lower.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameColumnDimension, "lower");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != (RowCount + lower.RowCount) || result.ColumnCount != ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
// Clear the result matrix
|
|
result.Clear();
|
|
|
|
// Copy the diagonal part into the result matrix.
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
|
|
// Copy the lower matrix into the result matrix.
|
|
for (var i = 0; i < lower.RowCount; i++)
|
|
{
|
|
for (var j = 0; j < lower.ColumnCount; j++)
|
|
{
|
|
result[i + RowCount, j] = lower[i, j];
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Concatenates this matrix with the given matrix.
|
|
/// </summary>
|
|
/// <param name="right">The matrix to concatenate.</param>
|
|
/// <returns>The combined <see cref="SparseMatrix"/>.</returns>
|
|
public override Matrix<Complex> Append(Matrix<Complex> right)
|
|
{
|
|
if (right == null)
|
|
{
|
|
throw new ArgumentNullException("right");
|
|
}
|
|
|
|
if (right.RowCount != RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameRowDimension);
|
|
}
|
|
|
|
var result = new SparseMatrix(RowCount, ColumnCount + right.ColumnCount);
|
|
Append(right, result);
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Concatenates this matrix with the given matrix and places the result into the result <see cref="SparseMatrix"/>.
|
|
/// </summary>
|
|
/// <param name="right">The matrix to concatenate.</param>
|
|
/// <param name="result">The combined <see cref="SparseMatrix"/>.</param>
|
|
public override void Append(Matrix<Complex> right, Matrix<Complex> result)
|
|
{
|
|
if (right == null)
|
|
{
|
|
throw new ArgumentNullException("right");
|
|
}
|
|
|
|
if (right.RowCount != RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameRowDimension);
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.ColumnCount != (ColumnCount + right.ColumnCount) || result.RowCount != RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixSameColumnDimension);
|
|
}
|
|
|
|
// Clear the result matrix
|
|
result.Clear();
|
|
|
|
// Copy the diagonal part into the result matrix.
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
|
|
// Copy the lower matrix into the result matrix.
|
|
for (var i = 0; i < right.RowCount; i++)
|
|
{
|
|
for (var j = 0; j < right.ColumnCount; j++)
|
|
{
|
|
result[i, j + RowCount] = right[i, j];
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Diagonally stacks his matrix on top of the given matrix. The new matrix is a M-by-N matrix,
|
|
/// where M = this.Rows + lower.Rows and N = this.Columns + lower.Columns.
|
|
/// The values of off the off diagonal matrices/blocks are set to zero.
|
|
/// </summary>
|
|
/// <param name="lower">The lower, right matrix.</param>
|
|
/// <exception cref="ArgumentNullException">If lower is <see langword="null" />.</exception>
|
|
/// <returns>the combined matrix</returns>
|
|
public override Matrix<Complex> DiagonalStack(Matrix<Complex> lower)
|
|
{
|
|
if (lower == null)
|
|
{
|
|
throw new ArgumentNullException("lower");
|
|
}
|
|
|
|
var result = new SparseMatrix(RowCount + lower.RowCount, ColumnCount + lower.ColumnCount);
|
|
DiagonalStack(lower, result);
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Diagonally stacks his matrix on top of the given matrix and places the combined matrix into the result matrix.
|
|
/// </summary>
|
|
/// <param name="lower">The lower, right matrix.</param>
|
|
/// <param name="result">The combined matrix</param>
|
|
/// <exception cref="ArgumentNullException">If lower is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If the result matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If the result matrix's dimensions are not (this.Rows + lower.rows) x (this.Columns + lower.Columns).</exception>
|
|
public override void DiagonalStack(Matrix<Complex> lower, Matrix<Complex> result)
|
|
{
|
|
if (lower == null)
|
|
{
|
|
throw new ArgumentNullException("lower");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (result.RowCount != RowCount + lower.RowCount || result.ColumnCount != ColumnCount + lower.ColumnCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
// Clear the result matrix
|
|
result.Clear();
|
|
|
|
// Copy the diagonal part into the result matrix.
|
|
for (var i = 0; i < Data.Length; i++)
|
|
{
|
|
result[i, i] = Data[i];
|
|
}
|
|
|
|
// Copy the lower matrix into the result matrix.
|
|
CommonParallel.For(0, lower.RowCount, i => CommonParallel.For(0, lower.ColumnCount, j => result.At(i + RowCount, j + ColumnCount, lower.At(i, j))));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Pointwise multiplies this matrix with another matrix and stores the result into the result matrix.
|
|
/// </summary>
|
|
/// <param name="other">The matrix to pointwise multiply with this one.</param>
|
|
/// <param name="result">The matrix to store the result of the pointwise multiplication.</param>
|
|
/// <exception cref="ArgumentNullException">If the other matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentNullException">If the result matrix is <see langword="null" />.</exception>
|
|
/// <exception cref="ArgumentException">If this matrix and <paramref name="other"/> are not the same size.</exception>
|
|
/// <exception cref="ArgumentException">If this matrix and <paramref name="result"/> are not the same size.</exception>
|
|
public override void PointwiseMultiply(Matrix<Complex> other, Matrix<Complex> result)
|
|
{
|
|
if (other == null)
|
|
{
|
|
throw new ArgumentNullException("other");
|
|
}
|
|
|
|
if (result == null)
|
|
{
|
|
throw new ArgumentNullException("result");
|
|
}
|
|
|
|
if (ColumnCount != other.ColumnCount || RowCount != other.RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
if (ColumnCount != result.ColumnCount || RowCount != result.RowCount)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMatrixDimensions, "result");
|
|
}
|
|
|
|
var m = other as DiagonalMatrix;
|
|
var r = result as DiagonalMatrix;
|
|
|
|
if (m == null || r == null)
|
|
{
|
|
base.PointwiseMultiply(other, result);
|
|
}
|
|
else
|
|
{
|
|
Control.LinearAlgebraProvider.PointWiseMultiplyArrays(Data, m.Data, r.Data);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Permute the columns of a matrix according to a permutation.
|
|
/// </summary>
|
|
/// <param name="p">The column permutation to apply to this matrix.</param>
|
|
/// <exception cref="InvalidOperationException">Always thrown</exception>
|
|
/// <remarks>Permutation in diagonal matrix are senseless, because of matrix nature</remarks>
|
|
public override void PermuteColumns(Permutation p)
|
|
{
|
|
throw new InvalidOperationException("Permutations in diagonal matrix are not allowed");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Permute the rows of a matrix according to a permutation.
|
|
/// </summary>
|
|
/// <param name="p">The row permutation to apply to this matrix.</param>
|
|
/// <exception cref="InvalidOperationException">Always thrown</exception>
|
|
/// <remarks>Permutation in diagonal matrix are senseless, because of matrix nature</remarks>
|
|
public override void PermuteRows(Permutation p)
|
|
{
|
|
throw new InvalidOperationException("Permutations in diagonal matrix are not allowed");
|
|
}
|
|
|
|
#region Static constructors for special matrices.
|
|
|
|
/// <summary>
|
|
/// Initializes a square <see cref="DiagonalMatrix"/> with all zero's except for ones on the diagonal.
|
|
/// </summary>
|
|
/// <param name="order">the size of the square matrix.</param>
|
|
/// <returns>A diagonal identity matrix.</returns>
|
|
/// <exception cref="ArgumentException">
|
|
/// If <paramref name="order"/> is less than one.
|
|
/// </exception>
|
|
public static DiagonalMatrix Identity(int order)
|
|
{
|
|
var m = new DiagonalMatrix(order);
|
|
for (var i = 0; i < order; i++)
|
|
{
|
|
m.Data[i] = 1.0;
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Negates each element of this matrix.
|
|
/// </summary>
|
|
public override void Negate()
|
|
{
|
|
Multiply(-1);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates matrix with random elements.
|
|
/// </summary>
|
|
/// <param name="numberOfRows">Number of rows.</param>
|
|
/// <param name="numberOfColumns">Number of columns.</param>
|
|
/// <param name="distribution">Continuous Random Distribution or Source</param>
|
|
/// <returns>
|
|
/// An <c>numberOfRows</c>-by-<c>numberOfColumns</c> matrix with elements distributed according to the provided distribution.
|
|
/// </returns>
|
|
/// <exception cref="ArgumentException">If the parameter <paramref name="numberOfRows"/> is not positive.</exception>
|
|
/// <exception cref="ArgumentException">If the parameter <paramref name="numberOfColumns"/> is not positive.</exception>
|
|
public override Matrix<Complex> Random(int numberOfRows, int numberOfColumns, IContinuousDistribution distribution)
|
|
{
|
|
if (numberOfRows < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "numberOfRows");
|
|
}
|
|
|
|
if (numberOfColumns < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "numberOfColumns");
|
|
}
|
|
|
|
var matrix = CreateMatrix(numberOfRows, numberOfColumns);
|
|
var mn = Math.Min(numberOfRows, numberOfColumns);
|
|
CommonParallel.For(0, mn, i => matrix[i, i] = new Complex(distribution.Sample(), distribution.Sample()));
|
|
|
|
return matrix;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Generates matrix with random elements.
|
|
/// </summary>
|
|
/// <param name="numberOfRows">Number of rows.</param>
|
|
/// <param name="numberOfColumns">Number of columns.</param>
|
|
/// <param name="distribution">Continuous Random Distribution or Source</param>
|
|
/// <returns>
|
|
/// An <c>numberOfRows</c>-by-<c>numberOfColumns</c> matrix with elements distributed according to the provided distribution.
|
|
/// </returns>
|
|
/// <exception cref="ArgumentException">If the parameter <paramref name="numberOfRows"/> is not positive.</exception>
|
|
/// <exception cref="ArgumentException">If the parameter <paramref name="numberOfColumns"/> is not positive.</exception>
|
|
public override Matrix<Complex> Random(int numberOfRows, int numberOfColumns, IDiscreteDistribution distribution)
|
|
{
|
|
if (numberOfRows < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "numberOfRows");
|
|
}
|
|
|
|
if (numberOfColumns < 1)
|
|
{
|
|
throw new ArgumentException(Resources.ArgumentMustBePositive, "numberOfColumns");
|
|
}
|
|
|
|
var matrix = CreateMatrix(numberOfRows, numberOfColumns);
|
|
var mn = Math.Min(numberOfRows, numberOfColumns);
|
|
CommonParallel.For(0, mn, i => matrix[i, i] = new Complex(distribution.Sample(), distribution.Sample()));
|
|
|
|
return matrix;
|
|
}
|
|
|
|
#region Simple arithmetic of type T
|
|
/// <summary>
|
|
/// Add two values T+T
|
|
/// </summary>
|
|
/// <param name="val1">Left operand value</param>
|
|
/// <param name="val2">Right operand value</param>
|
|
/// <returns>Result of addition</returns>
|
|
protected sealed override Complex AddT(Complex val1, Complex val2)
|
|
{
|
|
return val1 + val2;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Subtract two values T-T
|
|
/// </summary>
|
|
/// <param name="val1">Left operand value</param>
|
|
/// <param name="val2">Right operand value</param>
|
|
/// <returns>Result of subtract</returns>
|
|
protected sealed override Complex SubtractT(Complex val1, Complex val2)
|
|
{
|
|
return val1 - val2;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Multiply two values T*T
|
|
/// </summary>
|
|
/// <param name="val1">Left operand value</param>
|
|
/// <param name="val2">Right operand value</param>
|
|
/// <returns>Result of multiplication</returns>
|
|
protected sealed override Complex MultiplyT(Complex val1, Complex val2)
|
|
{
|
|
return val1 * val2;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Divide two values T/T
|
|
/// </summary>
|
|
/// <param name="val1">Left operand value</param>
|
|
/// <param name="val2">Right operand value</param>
|
|
/// <returns>Result of divide</returns>
|
|
protected sealed override Complex DivideT(Complex val1, Complex val2)
|
|
{
|
|
return val1 / val2;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Take absolute value
|
|
/// </summary>
|
|
/// <param name="val1">Source alue</param>
|
|
/// <returns>True if one; otherwise false</returns>
|
|
protected sealed override double AbsoluteT(Complex val1)
|
|
{
|
|
return val1.Magnitude;
|
|
}
|
|
#endregion
|
|
}
|
|
}
|
|
|