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.
227 lines
8.9 KiB
227 lines
8.9 KiB
// <copyright file="MatrixMarketWriter.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-2013 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>
|
|
|
|
using System;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.IO.Compression;
|
|
using System.Numerics;
|
|
using MathNet.Numerics.LinearAlgebra;
|
|
using MathNet.Numerics.LinearAlgebra.Storage;
|
|
|
|
namespace MathNet.Numerics.Data.Text
|
|
{
|
|
/// <summary>
|
|
/// NIST MatrixMarket Format Writer (http://math.nist.gov/MatrixMarket/)
|
|
/// </summary>
|
|
public static class MatrixMarketWriter
|
|
{
|
|
static readonly NumberFormatInfo Format = CultureInfo.InvariantCulture.NumberFormat;
|
|
|
|
public static void WriteMatrix<T>(string filePath, Matrix<T> matrix, bool compress = false) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
using (var stream = File.OpenWrite(filePath))
|
|
{
|
|
if (compress)
|
|
{
|
|
using (var compressed = new GZipStream(stream, CompressionMode.Compress))
|
|
using (var buffered = new BufferedStream(compressed, 4096))
|
|
using (var writer = new StreamWriter(buffered))
|
|
{
|
|
WriteMatrix(writer, matrix);
|
|
return;
|
|
}
|
|
}
|
|
using (var writer = new StreamWriter(stream))
|
|
{
|
|
WriteMatrix(writer, matrix);
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void WriteVector<T>(string filePath, Vector<T> vector, bool compress = false) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
using (var stream = File.OpenRead(filePath))
|
|
{
|
|
if (compress)
|
|
{
|
|
using (var compressed = new GZipStream(stream, CompressionMode.Compress))
|
|
using (var buffered = new BufferedStream(compressed, 4096))
|
|
using (var writer = new StreamWriter(buffered))
|
|
{
|
|
WriteVector(writer, vector);
|
|
return;
|
|
}
|
|
}
|
|
using (var writer = new StreamWriter(stream))
|
|
{
|
|
WriteVector(writer, vector);
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void WriteMatrix<T>(Stream stream, Matrix<T> matrix) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
using (var writer = new StreamWriter(stream))
|
|
{
|
|
WriteMatrix(writer, matrix);
|
|
}
|
|
}
|
|
|
|
public static void WriteVector<T>(Stream stream, Vector<T> vector) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
using (var writer = new StreamWriter(stream))
|
|
{
|
|
WriteVector(writer, vector);
|
|
}
|
|
}
|
|
|
|
public static void WriteMatrix<T>(TextWriter writer, Matrix<T> matrix) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
var complex = typeof (T) == typeof (Complex) || typeof (T) == typeof (Complex32);
|
|
var format = CreateValueFormatter<T>();
|
|
var storage = matrix.Storage;
|
|
|
|
var sparse = storage as SparseCompressedRowMatrixStorage<T>;
|
|
if (sparse != null)
|
|
{
|
|
writer.WriteLine("%%MatrixMarket matrix coordinate {0} general", complex ? "complex" : "real");
|
|
writer.WriteLine("{0} {1} {2}", sparse.RowCount, sparse.ColumnCount, sparse.ValueCount);
|
|
for (int row = 0; row < sparse.RowCount; row++)
|
|
{
|
|
var endIndex = sparse.RowPointers[row + 1];
|
|
for (var j = sparse.RowPointers[row]; j < endIndex; j++)
|
|
{
|
|
writer.WriteLine("{0} {1} {2}", row + 1, sparse.ColumnIndices[j] + 1, format(sparse.Values[j]));
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
var diagonal = storage as DiagonalMatrixStorage<T>;
|
|
if (diagonal != null)
|
|
{
|
|
writer.WriteLine("%%MatrixMarket matrix coordinate {0} general", complex ? "complex" : "real");
|
|
writer.WriteLine("{0} {1} {2}", diagonal.RowCount, diagonal.ColumnCount, diagonal.Data.Length);
|
|
for (int k = 0; k < diagonal.Data.Length; k++)
|
|
{
|
|
writer.WriteLine("{0} {1} {2}", k + 1, k + 1, format(diagonal.Data[k]));
|
|
}
|
|
return;
|
|
}
|
|
|
|
var dense = storage as DenseColumnMajorMatrixStorage<T>;
|
|
if (dense != null)
|
|
{
|
|
writer.WriteLine("%%MatrixMarket matrix array {0} general", complex ? "complex" : "real");
|
|
writer.WriteLine("{0} {1}", dense.RowCount, dense.ColumnCount);
|
|
foreach (var value in dense.Data)
|
|
{
|
|
writer.WriteLine(format(value));
|
|
}
|
|
return;
|
|
}
|
|
|
|
writer.WriteLine("%%MatrixMarket matrix array {0} general", complex ? "complex" : "real");
|
|
writer.WriteLine("{0} {1}", storage.RowCount, storage.ColumnCount);
|
|
foreach (var value in storage.ToColumnMajorArray())
|
|
{
|
|
writer.WriteLine(format(value));
|
|
}
|
|
}
|
|
|
|
public static void WriteVector<T>(TextWriter writer, Vector<T> vector) where T : struct, IEquatable<T>, IFormattable
|
|
{
|
|
var complex = typeof (T) == typeof (Complex) || typeof (T) == typeof (Complex32);
|
|
var format = CreateValueFormatter<T>();
|
|
var storage = vector.Storage;
|
|
|
|
var sparse = storage as SparseVectorStorage<T>;
|
|
if (sparse != null)
|
|
{
|
|
writer.WriteLine("%%MatrixMarket vector coordinate {0}", complex ? "complex" : "real");
|
|
writer.WriteLine("{0} {1}", sparse.Length, sparse.ValueCount);
|
|
for (var k = 0; k < sparse.ValueCount; k++)
|
|
{
|
|
writer.WriteLine("{0} {1}", k + 1, format(sparse.Values[k]));
|
|
}
|
|
return;
|
|
}
|
|
|
|
var dense = storage as DenseVectorStorage<T>;
|
|
if (dense != null)
|
|
{
|
|
writer.WriteLine("%%MatrixMarket vector array {0}", complex ? "complex" : "real");
|
|
writer.WriteLine("{0}", dense.Length);
|
|
foreach (var value in dense.Data)
|
|
{
|
|
writer.WriteLine(format(value));
|
|
}
|
|
return;
|
|
}
|
|
|
|
writer.WriteLine("%%MatrixMarket vector array {0}", complex ? "complex" : "real");
|
|
writer.WriteLine("{0}", storage.Length);
|
|
foreach (var value in storage.Enumerate())
|
|
{
|
|
writer.WriteLine(format(value));
|
|
}
|
|
}
|
|
|
|
static Func<T, string> CreateValueFormatter<T>() where T : IFormattable
|
|
{
|
|
if (typeof (T) == typeof (double))
|
|
{
|
|
return value => string.Format(Format, "{0:G14}", value);
|
|
}
|
|
if (typeof (T) == typeof (float))
|
|
{
|
|
return value => string.Format(Format, "{0:G7}", value);
|
|
}
|
|
if (typeof (T) == typeof (Complex))
|
|
{
|
|
return value =>
|
|
{
|
|
var c = (Complex) (object) value;
|
|
return string.Format(Format, "{0:G14} {1:G14}", c.Real, c.Imaginary);
|
|
};
|
|
}
|
|
if (typeof (T) == typeof (Complex32))
|
|
{
|
|
return value =>
|
|
{
|
|
var c = (Complex32) (object) value;
|
|
return string.Format(Format, "{0:G7} {1:G7}", c.Real, c.Imaginary);
|
|
};
|
|
}
|
|
throw new NotSupportedException();
|
|
}
|
|
}
|
|
}
|
|
|