/*
* SVM.NET Library
* Copyright (C) 2008 Matthew Johnson
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
using System;
using System.IO;
namespace SVM
{
///
/// Class which encapsulates a range transformation.
///
public class RangeTransform : IRangeTransform
{
private double[] _inputStart;
private double[] _inputScale;
private double _outputStart;
private double _outputScale;
private int _length;
///
/// Constructor.
///
/// The minimum values in each dimension.
/// The maximum values in each dimension.
/// The desired lower bound for all dimensions.
/// The desired upper bound for all dimensions.
public RangeTransform(double[] minValues, double[] maxValues, double lowerBound, double upperBound)
{
_length = minValues.Length;
if(maxValues.Length != _length)
throw new Exception("Number of max and min values must be equal.");
_inputStart = new double[_length];
_inputScale = new double[_length];
for (int i = 0; i < _length; i++)
{
_inputStart[i] = minValues[i];
_inputScale[i] = maxValues[i] - minValues[i];
}
_outputStart = lowerBound;
_outputScale = upperBound - lowerBound;
}
private RangeTransform(double[] inputStart, double[] inputScale, double outputStart, double outputScale, int length)
{
_inputStart = inputStart;
_inputScale = inputScale;
_outputStart = outputStart;
_outputScale = outputScale;
_length = length;
}
///
/// Transforms the input array based upon the values provided.
///
/// The input array
/// A scaled array
public Node[] Transform(Node[] input)
{
Node[] output = new Node[input.Length];
for (int i = 0; i < _length; i++)
{
int index = input[i].Index;
double value = input[i].Value;
output[i] = new Node(index, Transform(value, index));
}
return output;
}
///
/// Transforms this an input value using the scaling transform for the provided dimension.
///
/// The input value to transform
/// The dimension whose scaling transform should be used
/// The scaled value
public double Transform(double input, int index)
{
index--;
double tmp = input - _inputStart[index];
if (_inputScale[index] == 0)
return 0;
tmp /= _inputScale[index];
tmp *= _outputScale;
return tmp + _outputStart;
}
///
/// Writes this Range transform to a stream.
///
/// The stream to write to
/// The range to write
public static void Write(Stream stream, RangeTransform r)
{
StreamWriter output = new StreamWriter(stream);
output.WriteLine(r._length);
output.Write(r._inputStart[0]);
for(int i=1; i
/// Writes this Range transform to a file. This will overwrite any previous data in the file.
///
/// The file to write to
/// The Range to write
public static void Write(string outputFile, RangeTransform r)
{
FileStream s = File.Open(outputFile, FileMode.Create);
try
{
Write(s, r);
}
finally
{
s.Close();
}
}
///
/// Reads a Range transform from a file.
///
/// The file to read from
/// The Range transform
public static RangeTransform Read(string inputFile)
{
FileStream s = File.OpenRead(inputFile);
try
{
return Read(s);
}
finally
{
s.Close();
}
}
///
/// Reads a Range transform from a stream.
///
/// The stream to read from
/// The Range transform
public static RangeTransform Read(Stream stream)
{
StreamReader input = new StreamReader(stream);
int length = int.Parse(input.ReadLine());
double[] inputStart = new double[length];
double[] inputScale = new double[length];
string[] parts = input.ReadLine().Split();
for (int i = 0; i < length; i++)
inputStart[i] = double.Parse(parts[i]);
parts = input.ReadLine().Split();
for (int i = 0; i < length; i++)
inputScale[i] = double.Parse(parts[i]);
parts = input.ReadLine().Split();
double outputStart = double.Parse(parts[0]);
double outputScale = double.Parse(parts[1]);
return new RangeTransform(inputStart, inputScale, outputStart, outputScale, length);
}
}
}