Math.NET Numerics
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.
 
 
 

507 lines
17 KiB

// <copyright file="VectorTests.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.UnitTests.LinearAlgebraTests.Complex
{
using System;
using System.Collections;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Numerics;
using Distributions;
using LinearAlgebra.Generic;
using NUnit.Framework;
/// <summary>
/// Abstract class with the common set of vector tests.
/// </summary>
[TestFixture]
public abstract partial class VectorTests
{
/// <summary>
/// Test vector values.
/// </summary>
protected readonly Complex[] Data = { new Complex(1, 1), new Complex(2, 1), new Complex(3, 1), new Complex(4, 1), new Complex(5, 1) };
/// <summary>
/// Can clone a vector.
/// </summary>
[Test]
public void CanCloneVector()
{
var vector = CreateVector(Data);
var clone = vector.Clone();
Assert.AreNotSame(vector, clone);
Assert.AreEqual(vector.Count, clone.Count);
CollectionAssert.AreEqual(vector, clone);
}
/// <summary>
/// Can clone a vector using <c>IClonable</c> interface method.
/// </summary>
[Test]
public void CanCloneVectorUsingICloneable()
{
var vector = CreateVector(Data);
var clone = (Vector<Complex>)((ICloneable)vector).Clone();
Assert.AreNotSame(vector, clone);
Assert.AreEqual(vector.Count, clone.Count);
CollectionAssert.AreEqual(vector, clone);
}
/// <summary>
/// Can convert vector to string.
/// </summary>
[Test]
public void CanConvertVectorToString()
{
var vector = CreateVector(Data);
var str = vector.ToString();
var sep = CultureInfo.CurrentCulture.TextInfo.ListSeparator;
Assert.AreEqual(string.Format("(1{0} 1){1}(2{0} 1){1}(3{0} 1){1}(4{0} 1){1}(5{0} 1)", ",", sep), str);
}
/// <summary>
/// Can copy part of a vector to another vector.
/// </summary>
[Test]
public void CanCopyPartialVectorToAnother()
{
var vector = CreateVector(Data);
var other = CreateVector(Data.Length);
vector.CopyTo(other, 2, 2, 2);
AssertHelpers.AreEqual(Complex.Zero, other[0]);
AssertHelpers.AreEqual(Complex.Zero, other[1]);
AssertHelpers.AreEqual(new Complex(3, 1), other[2]);
AssertHelpers.AreEqual(new Complex(4, 1), other[3]);
AssertHelpers.AreEqual(Complex.Zero, other[4]);
}
/// <summary>
/// Can copy part of a vector to the same vector.
/// </summary>
[Test]
public void CanCopyPartialVectorToSelf()
{
var vector = CreateVector(Data);
vector.CopyTo(vector, 0, 2, 2);
AssertHelpers.AreEqual(new Complex(1, 1), vector[0]);
AssertHelpers.AreEqual(new Complex(2, 1), vector[1]);
AssertHelpers.AreEqual(new Complex(1, 1), vector[2]);
AssertHelpers.AreEqual(new Complex(2, 1), vector[3]);
AssertHelpers.AreEqual(new Complex(5, 1), vector[4]);
}
/// <summary>
/// Can copy a vector to another vector.
/// </summary>
[Test]
public void CanCopyVectorToAnother()
{
var vector = CreateVector(Data);
var other = CreateVector(Data.Length);
vector.CopyTo(other);
CollectionAssert.AreEqual(vector, other);
}
/// <summary>
/// Can create a matrix using instance of a vector.
/// </summary>
[Test]
public void CanCreateMatrix()
{
var vector = CreateVector(Data);
var matrix = vector.CreateMatrix(10, 10);
Assert.AreEqual(matrix.RowCount, 10);
Assert.AreEqual(matrix.ColumnCount, 10);
}
/// <summary>
/// Can create a vector using the instance of vector.
/// </summary>
[Test]
public void CanCreateVector()
{
var expected = CreateVector(5);
var actual = expected.CreateVector(5);
Assert.AreEqual(expected.GetType(), actual.GetType(), "vectors are same type.");
}
/// <summary>
/// Can enumerate over a vector.
/// </summary>
[Test]
public void CanEnumerateOverVector()
{
var vector = CreateVector(Data);
for (var i = 0; i < Data.Length; i++)
{
Assert.AreEqual(Data[i], vector[i]);
}
}
/// <summary>
/// Can enumerate over a vector using <c>IEnumerable</c> interface.
/// </summary>
[Test]
public void CanEnumerateOverVectorUsingIEnumerable()
{
var enumerable = (IEnumerable)CreateVector(Data);
var index = 0;
foreach (var element in enumerable)
{
Assert.AreEqual(Data[index++], (Complex)element);
}
}
/// <summary>
/// Can equate vectors.
/// </summary>
[Test]
public void CanEquateVectors()
{
var vector1 = CreateVector(Data);
var vector2 = CreateVector(Data);
var vector3 = CreateVector(4);
Assert.IsTrue(vector1.Equals(vector1));
Assert.IsTrue(vector1.Equals(vector2));
Assert.IsFalse(vector1.Equals(vector3));
Assert.IsFalse(vector1.Equals(null));
Assert.IsFalse(vector1.Equals(2));
}
/// <summary>
/// <c>CreateVector</c> throws <c>ArgumentException</c> if size is not positive.
/// </summary>
[Test]
public void SizeIsNotPositiveThrowsArgumentException()
{
Assert.Throws<ArgumentException>(() => CreateVector(-1));
Assert.Throws<ArgumentException>(() => CreateVector(0));
}
/// <summary>
/// Can equate vectors using Object.Equals.
/// </summary>
[Test]
public void CanEquateVectorsUsingObjectEquals()
{
var vector1 = CreateVector(Data);
var vector2 = CreateVector(Data);
Assert.IsTrue(vector1.Equals((object)vector2));
}
/// <summary>
/// Can get hash code of a vector.
/// </summary>
[Test]
public void CanGetHashCode()
{
var vector = CreateVector(new[] { new Complex(1, 1), new Complex(2, 1), new Complex(3, 1), new Complex(4, 1), new Complex(5, 1) });
Assert.AreEqual(1104007323, vector.GetHashCode());
}
/// <summary>
/// Can enumerate over a vector using indexed enumerator.
/// </summary>
[Test]
public void CanEnumerateOverVectorUsingIndexedEnumerator()
{
var vector = CreateVector(Data);
foreach (var pair in vector.GetIndexedEnumerator())
{
Assert.AreEqual(Data[pair.Item1], pair.Item2);
}
}
/// <summary>
/// Can convert a vector to array.
/// </summary>
[Test]
public void CanConvertVectorToArray()
{
var vector = CreateVector(Data);
var array = vector.ToArray();
CollectionAssert.AreEqual(array, vector);
}
/// <summary>
/// Can convert a vector to column matrix.
/// </summary>
[Test]
public void CanConvertVectorToColumnMatrix()
{
var vector = CreateVector(Data);
var matrix = vector.ToColumnMatrix();
Assert.AreEqual(vector.Count, matrix.RowCount);
Assert.AreEqual(1, matrix.ColumnCount);
for (var i = 0; i < vector.Count; i++)
{
Assert.AreEqual(vector[i], matrix[i, 0]);
}
}
/// <summary>
/// Can convert a vector to row matrix.
/// </summary>
[Test]
public void CanConvertVectorToRowMatrix()
{
var vector = CreateVector(Data);
var matrix = vector.ToRowMatrix();
Assert.AreEqual(vector.Count, matrix.ColumnCount);
Assert.AreEqual(1, matrix.RowCount);
for (var i = 0; i < vector.Count; i++)
{
Assert.AreEqual(vector[i], matrix[0, i]);
}
}
/// <summary>
/// Can set values in vector.
/// </summary>
[Test]
public void CanSetValues()
{
var vector = CreateVector(Data);
vector.SetValues(Data);
CollectionAssert.AreEqual(vector, Data);
}
/// <summary>
/// Can get subvector from a vector.
/// </summary>
/// <param name="index">The first element to begin copying from.</param>
/// <param name="length">The number of elements to copy.</param>
[Test, Sequential]
public void CanGetSubVector([Values(0, 2, 1)] int index, [Values(5, 2, 4)] int length)
{
var vector = CreateVector(Data);
var sub = vector.SubVector(index, length);
Assert.AreEqual(length, sub.Count);
for (var i = 0; i < length; i++)
{
Assert.AreEqual(vector[i + index], sub[i]);
}
}
/// <summary>
/// Getting subvector using wrong parameters throw an exception.
/// </summary>
/// <param name="index">The first element to begin copying from.</param>
/// <param name="length">The number of elements to copy.</param>
[Test, Combinatorial]
public void CanGetSubVectorWithWrongValuesShouldThrowException([Values(6, 1)] int index, [Values(10, -10)] int length)
{
var vector = CreateVector(Data);
Assert.Throws<ArgumentOutOfRangeException>(() => vector.SubVector(index, length));
}
/// <summary>
/// Can find absolute minimum value index.
/// </summary>
[Test]
public void CanFindAbsoluteMinimumIndex()
{
var source = CreateVector(Data);
const int Expected = 0;
var actual = source.AbsoluteMinimumIndex();
Assert.AreEqual(Expected, actual);
}
/// <summary>
/// Can find absolute minimum value of a vector.
/// </summary>
[Test]
public void CanFindAbsoluteMinimum()
{
var source = CreateVector(Data);
var expected = new Complex(1, 1).Magnitude;
var actual = source.AbsoluteMinimum();
Assert.AreEqual(expected, actual.Real);
}
/// <summary>
/// Can find absolute maximum value index.
/// </summary>
[Test]
public void CanFindAbsoluteMaximumIndex()
{
var source = CreateVector(Data);
const int Expected = 4;
var actual = source.AbsoluteMaximumIndex();
Assert.AreEqual(Expected, actual);
}
/// <summary>
/// Can find absolute maximum value of a vector.
/// </summary>
[Test]
public void CanFindAbsoluteMaximum()
{
var source = CreateVector(Data);
var expected = new Complex(5, 1).Magnitude;
var actual = source.AbsoluteMaximum();
Assert.AreEqual(expected, actual.Real);
}
/// <summary>
/// Find maximum value index throws <c>NotSupportedException</c>.
/// </summary>
[Test]
public void FindMaximumIndexThrowsNotSupportedException()
{
var vector = CreateVector(Data);
Assert.Throws<NotSupportedException>(() => { var actual = vector.MaximumIndex(); });
}
/// <summary>
/// Find maximum value throws <c>NotSupportedException</c>.
/// </summary>
[Test]
public void FindMaximumThrowsNotSupportedException()
{
var vector = CreateVector(Data);
Assert.Throws<NotSupportedException>(() => { var actual = vector.Maximum(); });
}
/// <summary>
/// Find minimum value index throws <c>NotSupportedException</c>.
/// </summary>
[Test]
public void FindMinimumIndexThrowsNotSupportedException()
{
var vector = CreateVector(Data);
Assert.Throws<NotSupportedException>(() => { var actual = vector.MinimumIndex(); });
}
/// <summary>
/// Find minimum value throws <c>NotSupportedException</c>.
/// </summary>
[Test]
public void FindMinimumThrowsNotSupportedException()
{
var vector = CreateVector(Data);
Assert.Throws<NotSupportedException>(() => { var actual = vector.Minimum(); });
}
/// <summary>
/// Can compute the sum of a vector elements.
/// </summary>
[Test]
public void CanSum()
{
Complex[] testData = { new Complex(-20, -1), new Complex(-10, -1), new Complex(10, 1), new Complex(20, 1), new Complex(30, 1) };
var vector = CreateVector(testData);
var actual = vector.Sum();
var expected = new Complex(30, 1);
Assert.AreEqual(expected, actual);
}
/// <summary>
/// Can compute the sum of the absolute value a vector elements.
/// </summary>
[Test]
public void CanSumMagnitudes()
{
Complex[] testData = { new Complex(-20, -1), new Complex(-10, -1), new Complex(10, 1), new Complex(20, 1), new Complex(30, 1) };
var vector = CreateVector(testData);
var actual = vector.SumMagnitudes();
var expected = testData.Sum(complex => complex.Magnitude);
Assert.AreEqual(expected, actual.Real);
}
/// <summary>
/// Set values with <c>null</c> parameter throw exception.
/// </summary>
[Test]
public void SetValuesWithNullParameterThrowsArgumentException()
{
var vector = CreateVector(Data);
Assert.Throws<ArgumentNullException>(() => vector.SetValues(null));
}
/// <summary>
/// Set values with non-equal data length throw exception.
/// </summary>
[Test]
public void SetValuesWithNonEqualDataLengthThrowsArgumentException()
{
var vector = CreateVector(Data.Length + 2);
Assert.Throws<ArgumentException>(() => vector.SetValues(Data));
}
/// <summary>
/// Generate a vector with number of elements less than zero throw an exception.
/// </summary>
[Test]
public void RandomWithNumberOfElementsLessThanZeroThrowsArgumentException()
{
var vector = CreateVector(4);
Assert.Throws<ArgumentException>(() => vector.Random(-2, new ContinuousUniform()));
}
/// <summary>
/// Can clear a vector.
/// </summary>
[Test]
public void CanClearVector()
{
Complex[] testData = { new Complex(-20, -1), new Complex(-10, -1), new Complex(10, 1), new Complex(20, 1), new Complex(30, 1) };
var vector = CreateVector(testData);
vector.Clear();
foreach (var element in vector)
{
Assert.AreEqual(Complex.Zero, element);
}
}
/// <summary>
/// Creates a new instance of the Vector class.
/// </summary>
/// <param name="size">The size of the <strong>Vector</strong> to construct.</param>
/// <returns>The new <c>Vector</c>.</returns>
protected abstract Vector<Complex> CreateVector(int size);
/// <summary>
/// Creates a new instance of the Vector class.
/// </summary>
/// <param name="data">The array to create this vector from.</param>
/// <returns>The new <c>Vector</c>.</returns>
protected abstract Vector<Complex> CreateVector(IList<Complex> data);
}
}