All Downloads are FREE. Search and download functionalities are using the official Maven repository.

weka.estimators.UnivariateMixtureEstimator Maven / Gradle / Ivy

/*
 *   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 .
 */

/*
 *    UnivariateMixtureEstimator.java
 *    Copyright (C) 2014 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.estimators;

import java.io.Serializable;
import java.util.Arrays;
import java.util.Enumeration;
import java.util.Random;
import java.util.ArrayList;
import java.util.Vector;

import weka.core.ContingencyTables;
import weka.core.Option;
import weka.core.OptionHandler;
import weka.core.RevisionUtils;
import weka.core.Statistics;
import weka.core.Utils;

/**
 * Simple weighted mixture density estimator. Uses a mixture of Gaussians
 * and applies the leave-one-out bootstrap for model selection. Can alternatively use normalized entropy.
 *
 * @author Eibe Frank ([email protected])
 * @version $Revision: 8034 $
 */
public class UnivariateMixtureEstimator implements UnivariateDensityEstimator,
UnivariateIntervalEstimator,
UnivariateQuantileEstimator,
OptionHandler,
Serializable {

  /** Constant for normal distribution. */
  private static double m_normConst = Math.log(Math.sqrt(2 * Math.PI));

  /**
   * Fast univariate mixture model implementation.
   */
  public class MM {

    /** Means */
    protected double[] m_Means = null;

    /** Standard deviations */
    protected double[] m_StdDevs = null;

    /** The priors, on log scale */
    protected double[] m_LogPriors = null;

    /** The number of actual components */
    protected int m_K;

    /**
     * Returns string describing the estimator.
     */
    public String toString() {

      StringBuffer sb = new StringBuffer();
      sb.append("Mixture model estimator\n\n");
      for (int i = 0; i < m_LogPriors.length; i++) {
        sb.append("Mean: " + m_Means[i] + "\tStd. dev.: " + m_StdDevs[i] + "\tPrior prob.: " +
          Math.exp(m_LogPriors[i]) + "\n");
      }

      return sb.toString();
    }

    /**
     * Returns smallest distance to given elements.
     * Assumes m_Means has at least one element (i.e. m_K >= 1).
     */
    protected double smallestDistance(double val) {

      double min = Math.abs(val - m_Means[0]);
      for (int i = 1; i < m_K; i++) {
        if (Math.abs(val - m_Means[i]) < min) {
          min = Math.abs(val - m_Means[i]);
        }
      }
      return min;
    }

    /**
     * Returns the index of the nearest mean.
     * Assumes m_Means has at least one element (i.e. m_K >= 1).
     */
    protected int nearestMean(double val) {

      double min = Math.abs(val - m_Means[0]);
      int index = 0;
      for (int i = 1; i < m_K; i++) {
        if (Math.abs(val - m_Means[i]) < min) {
          min = Math.abs(val - m_Means[i]);
          index = i;
        }
      }
      return index;
    }

    /**
     * Initializes the model. Assumes K >= 1, values.length >= 1,
     * and values.length = weights.length.
     */
    public void initializeModel(int K, double[] values, double[] weights, Random r) {

      // Initialize means using farthest points
      m_Means = new double[K];

      // Randomly choose first point
      double furthestVal = values[r.nextInt(values.length)];

      // Find K maximally distant points (if possible)
      m_K = 0;
      do {
        m_Means[m_K] = furthestVal;
        m_K++;
        if (m_K >= K) {
          break;
        }
        double maxMinDist = smallestDistance(values[0]);
        furthestVal = values[0];
        for (int i = 1; i < values.length; i++) {
          double minDist = smallestDistance(values[i]);
          if (minDist > maxMinDist) {
            maxMinDist = minDist;
            furthestVal = values[i];
          }
        }
        if (maxMinDist <= 0) {
          break;
        }
      } while (true);

      // Shrink array of means if necessary
      if (m_K < K) {
        double[] tempMeans = new double[m_K];
        System.arraycopy(m_Means, 0, tempMeans, 0, m_K);
        m_Means = tempMeans;
      }

      // Establish initial cluster assignments
      double[][] probs = new double[m_K][values.length];
      for (int i = 0; i < values.length; i++) {
        probs[nearestMean(values[i])][i] = 1.0;
      }

      // Compute initial parameters
      m_StdDevs = new double[m_K];
      m_LogPriors = new double[m_K];
      estimateParameters(values, weights, probs);
    }

    /**
     * Estimate parameters.
     */
    protected void estimateParameters(double[] values, double[] weights, double[][] probs) {

      double totalSumOfWeights = 0;
      for (int j = 0; j < m_K; j++) {
        double sum = 0;
        double sumWeights = 0;
        for (int i = 0; i < values.length; i++) {
          double weight = probs[j][i] * weights[i];
          sum += weight * values[i];
          sumWeights += weight;
        }
        if (sumWeights <= 0) {
          m_Means[j] = 0;
        } else {
          m_Means[j] = sum / sumWeights;
        }
        totalSumOfWeights += sumWeights;
      }

      for (int j = 0; j < m_K; j++) {
        double sum = 0;
        double sumWeights = 0;
        for (int i = 0; i < values.length; i++) {
          double weight = probs[j][i] * weights[i];
          double diff = values[i] - m_Means[j];
          sum += weight * diff * diff;
          sumWeights += weight;
        }
        if ((sum <= 0) || (sumWeights <= 0)) {
          m_StdDevs[j] = 1.0e-6; // Hack to prevent unpleasantness
        } else {
          m_StdDevs[j] = Math.sqrt(sum / sumWeights);
          if (m_StdDevs[j] < 1.0e-6) {
            m_StdDevs[j] = 1.0e-6;
          }
        }
        if (sumWeights <= 0) {
          m_LogPriors[j] = -Double.MAX_VALUE;
        } else {
          m_LogPriors[j] = Math.log(sumWeights / totalSumOfWeights);
        }
      }
    }

    /**
     * Computes loglikelihood of current model.
     */
    public double loglikelihood(double[] values, double[] weights) {

      double sum = 0;
      double sumOfWeights = 0;
      for (int i = 0; i < values.length; i++) {
        sum += weights[i] * logDensity(values[i]);
        sumOfWeights += weights[i];
      }
      return sum / sumOfWeights;
    }

    /**
     * Returns average of squared errors for current model.
     */
    public double MSE() {

      double mse = 0;
      for (int i = 0; i < m_K; i++) {
        mse += m_StdDevs[i] * m_StdDevs[i] * Math.exp(m_LogPriors[i]);
      }
      return mse;
    }  

    /**
     * Density function of normal distribution.
     */
    protected double logNormalDens(double x, double mean, double stdDev) {

      double diff = x - mean;
      return -(diff * diff / (2 * stdDev * stdDev)) - m_normConst - Math.log(stdDev);
    }

    /**
     * Joint densities per cluster.
     */
    protected double[] logJointDensities(double value) {

      double[] a = new double[m_K];
      for (int i = 0; i < m_K; i++) {
        a[i] = m_LogPriors[i] + logNormalDens(value, m_Means[i], m_StdDevs[i]);
      }
      return a;
    }

    /**
     * Computes log of density for given value.
     */
    public double logDensity(double value) {

      double[] a = logJointDensities(value);
      double max = a[Utils.maxIndex(a)];
      double sum = 0.0;
      for(int i = 0; i < a.length; i++) {
        sum += Math.exp(a[i] - max);
      }

      return max + Math.log(sum);
    }

    /**
     * Returns the interval for the given confidence value. 
     * 
     * @param conf the confidence value in the interval [0, 1]
     * @return the interval
     */
    public double[][] predictIntervals(double conf) {

      // Compute minimum and maximum value, and delta
      double val = Statistics.normalInverse(1.0 - (1.0 - conf) / 2);
      double min = Double.MAX_VALUE;
      double max = -Double.MAX_VALUE;
      for (int i = 0; i < m_Means.length; i++) {
        double l = m_Means[i] - val * m_StdDevs[i];
        if (l < min) {
          min = l;
        }
        double r = m_Means[i] + val * m_StdDevs[i];
        if (r > max) {
          max = r;
        }
      }
      double delta = (max - min) / m_NumIntervals;

      // Create array with estimated probabilities
      double[] probabilities = new double[m_NumIntervals];
      double leftVal = Math.exp(logDensity(min));
      for (int i = 0; i < m_NumIntervals; i++) {
        double rightVal = Math.exp(logDensity(min + (i + 1) * delta));
        probabilities[i] = 0.5 * (leftVal + rightVal) * delta;
        leftVal = rightVal;
      }

      // Sort array based on area of bin estimates
      int[] sortedIndices = Utils.sort(probabilities);

      // Mark the intervals to use
      double sum = 0;
      boolean[] toUse = new boolean[probabilities.length];
      int k = 0;
      while ((sum < conf) && (k < toUse.length)) {
        toUse[sortedIndices[toUse.length - (k + 1)]] = true;
        sum += probabilities[sortedIndices[toUse.length - (k + 1)]];
        k++;
      }

      // Don't need probabilities anymore
      probabilities = null;

      // Create final list of intervals
      ArrayList intervals = new ArrayList();

      // The current interval
      double[] interval = null;

      // Calculate actual intervals
      boolean haveStartedInterval = false;
      for (int i = 0; i < m_NumIntervals; i++) {

        // Should the current bin be used?
        if (toUse[i]) {

          // Do we need to create a new interval?
          if (haveStartedInterval == false) {
            haveStartedInterval = true;
            interval = new double[2];
            interval[0] = min + i * delta;
          }

          // Regardless, we should update the upper boundary
          interval[1] = min + (i + 1) * delta;
        } else {

          // We need to finalize and store the last interval
          // if necessary.
          if (haveStartedInterval) {
            haveStartedInterval = false;
            intervals.add(interval);
          }
        }
      }

      // Add last interval if there is one
      if (haveStartedInterval) {
        intervals.add(interval);
      }

      return intervals.toArray(new double[0][0]);
    }

    /**
     * Returns the quantile for the given percentage.
     * 
     * @param percentage the percentage
     * @return the quantile
     */
    public double predictQuantile(double percentage) {

      // Compute minimum and maximum value, and delta
      double valRight = Statistics.normalInverse(percentage);
      double valLeft = Statistics.normalInverse(0.001);
      double min = Double.MAX_VALUE;
      double max = -Double.MAX_VALUE;
      for (int i = 0; i < m_Means.length; i++) {
        double l = m_Means[i] - valLeft * m_StdDevs[i];
        if (l < min) {
          min = l;
        }
        double r = m_Means[i] + valRight * m_StdDevs[i];
        if (r > max) {
          max = r;
        }
      }
      double delta = (max - min) / m_NumIntervals;

      double sum = 0;
      double leftVal = Math.exp(logDensity(min));
      for (int i = 0; i < m_NumIntervals; i++) {
        if (sum >= percentage) {
          return min + i * delta;
        }
        double rightVal = Math.exp(logDensity(min + (i + 1) * delta));
        sum += 0.5 * (leftVal + rightVal) * delta;
        leftVal = rightVal;
      }
      return max;
    }
  }

  /** For serialization */
  private static final long serialVersionUID = -2035274930137353656L;

  /** The values used for this estimator */
  protected double[] m_Values = new double[1000];

  /** The weights used for this estimator */
  protected double[] m_Weights = new double[1000];

  /** The number of values that have been seen */
  protected int m_NumValues;

  /** The current mixture model */
  protected MM m_MixtureModel;

  /** The number of components to use (default is -1)*/
  protected int m_NumComponents = -1; 

  /** The maximum number of components to use (default is 5) */
  protected int m_MaxNumComponents = 5;

  /** The random number seed to use (default is 1*/
  protected int m_Seed = 1;

  /** The number of Bootstrap runs to use to select the number of components (default is 10) */
  protected int m_NumBootstrapRuns = 10;

  /** The number of intervals used to approximate prediction interval. */
  protected int m_NumIntervals = 1000;

  /** Whether to use normalized entropy instance of bootstrap. */
  protected boolean m_UseNormalizedEntropy = false;

  /** Whether to output debug info. */
  protected boolean m_Debug = false;

  /** The random number generator. */
  protected Random m_Random = new Random(m_Seed);

  /**
   * Returns a string describing the estimator.
   */
  public String globalInfo() {
    return "Estimates a univariate mixture model.";
  }

  /**
   * @return whether normalized entropy is used
   */
  public boolean getUseNormalizedEntropy() {
    return m_UseNormalizedEntropy;
  }

  /**
   * @param useNormalizedEntropy whether to use normalized entropy
   */
  public void setUseNormalizedEntropy(boolean useNormalizedEntropy) {
    m_UseNormalizedEntropy = useNormalizedEntropy;
  }

  /**
   * The tool tip for this property.
   */
  public String numBootstrapRunsToolTipText() {
    return "The number of Bootstrap runs to choose the number of components.";
  }

  /**
   * Returns the number of Bootstrap runs.
   * 
   * @return the number of Bootstrap runs
   */
  public int getNumBootstrapRuns() {
    return m_NumBootstrapRuns;
  }

  /**
   * Sets the number of Bootstrap runs.
   * 
   * @param mnumBootstrapRuns the number of Bootstrap runs
   */
  public void setNumBootstrapRuns(int numBootstrapRuns) {
    m_NumBootstrapRuns = numBootstrapRuns;
  }

  /**
   * The tool tip for this property.
   */
  public String numComponentsToolTipText() {
    return "The number of mixture components to use.";
  }

  /**
   * Returns the number of components to use.
   * 
   * @return the m_NumComponents
   */
  public int getNumComponents() {
    return m_NumComponents;
  }

  /**
   * Sets the number of components to use.
   * 
   * @param m_NumComponents the m_NumComponents to set
   */
  public void setNumComponents(int numComponents) {
    this.m_NumComponents = numComponents;
  }

  /**
   * Returns the tip text for this property
   * @return tip text for this property suitable for
   * displaying in the explorer/experimenter gui
   */
  public String seedTipText() {
    return "The random number seed to be used.";
  }

  /**
   * Set the seed for random number generation.
   *
   * @param seed the seed
   */
  public void setSeed(int seed) {

    m_Seed = seed;
    m_Random = new Random(seed);
  }

  /**
   * Gets the seed for the random number generations
   *
   * @return the seed for the random number generation
   */
  public int getSeed() {

    return m_Seed;
  } 
  /**
   * The tool tip for this property.
   */
  public String maxNumComponentsToolTipText() {
    return "The maximum number of mixture components to use.";
  }

  /**
   * Returns the number of components to use.
   * 
   * @return the maximum number of components to use
   */
  public int getMaxNumComponents() {
    return m_MaxNumComponents;
  }

  /**
   * Sets the number of components to use.
   * 
   * @param maxNumComponents the maximum number of components to evaluate
   */
  public void setMaxNumComponents(int maxNumComponents) {
    m_MaxNumComponents = maxNumComponents;
  }

  /**
   * Adds a value to the density estimator.
   *
   * @param value the value to add
   * @param weight the weight of the value
   */
  public void addValue(double value, double weight) {

    // Do we need to add value at all?
    if (!Utils.eq(weight, 0)) {

      // Invalidate current model
      m_MixtureModel = null;

      // Do we need to expand the arrays?
      if (m_NumValues == m_Values.length) {
        double[] newWeights = new double[2 * m_NumValues];
        double[] newValues = new double[2 * m_NumValues];
        System.arraycopy(m_Values, 0, newValues, 0, m_NumValues);
        System.arraycopy(m_Weights, 0, newWeights, 0, m_NumValues);
        m_Values = newValues;
        m_Weights = newWeights;
      }

      // Add values
      m_Values[m_NumValues] = value;
      m_Weights[m_NumValues] = weight;
      m_NumValues++;
    }
  }

  /**
   * Build mixture model. Assumes K >= 1, values.length >= 1,
   * and values.length = weights.length.
   */
  public MM buildModel(int K, double[] values, double[] weights) {

    // Initialize model using k-means
    MM model = null;
    double bestMSE = Double.MAX_VALUE;
    int numAttempts = 0;
    while (numAttempts < 5) {

      // Initialize model
      MM tempModel = new UnivariateMixtureEstimator().new MM();
      tempModel.initializeModel(K, values, weights, m_Random);

      // Run k-means until MSE converges
      double oldMSE = Double.MAX_VALUE;
      double MSE = tempModel.MSE();

      if (m_Debug) {
        System.err.println("MSE: " + MSE);
      }

      double[][] probs = new double[tempModel.m_K][values.length];
      while (Utils.sm(MSE, oldMSE)){

        // Compute memberships
        for (int j = 0; j < probs.length; j++) {
          Arrays.fill(probs[j], 0);
        }
        for (int i = 0; i < values.length; i++) {
          probs[tempModel.nearestMean(values[i])][i] = 1.0;
        }

        // Estimate parameters
        tempModel.estimateParameters(values, weights, probs);

        // Compute MSE for updated model
        oldMSE = MSE;
        MSE = tempModel.MSE();

        if (m_Debug) {
          System.err.println("MSE: " + MSE);
        }
      }
      if (MSE < bestMSE) {
        bestMSE = MSE;
        model = tempModel;
      }

      if (m_Debug) {
        System.err.println("Best MSE: " + bestMSE);
      }

      numAttempts++;
    }

    // Run until likelihood converges
    double oldLogLikelihood = -Double.MAX_VALUE;
    double loglikelihood = model.loglikelihood(values, weights);
    double[][] probs = new double[model.m_K][values.length];
    while (Utils.gr(loglikelihood, oldLogLikelihood)){

      // Establish membership probabilities
      for (int i = 0; i < values.length; i++) {
        double[] p = Utils.logs2probs(model.logJointDensities(values[i]));
        for (int j = 0; j < p.length; j++) {
          probs[j][i] = p[j];
        }
      }

      // Estimate parameters
      model.estimateParameters(values, weights, probs);

      // Compute loglikelihood for updated model
      oldLogLikelihood = loglikelihood;
      loglikelihood = model.loglikelihood(values, weights);
    }

    return model;
  }

  /**
   * Creates a new dataset of the same size using random sampling with
   * replacement according to the given weight vector. The weights of the
   * instances in the new dataset are set to one. 
   */
  public double[][] resampleWithWeights(Random random, boolean[] sampled) {

    // Walker's method, see pp. 232 of "Stochastic Simulation" by B.D. Ripley
    double[] P = new double[m_Weights.length];
    System.arraycopy(m_Weights, 0, P, 0, m_Weights.length);
    Utils.normalize(P);
    double[] Q = new double[m_Weights.length];
    int[] A = new int[m_Weights.length];
    int[] W = new int[m_Weights.length];
    int M = m_Weights.length;
    int NN = -1;
    int NP = M;
    for (int I = 0; I < M; I++) {
      if (P[I] < 0) {
        throw new IllegalArgumentException("Weights have to be positive.");
      }
      Q[I] = M * P[I];
      if (Q[I] < 1.0) {
        W[++NN] = I;
      } else {
        W[--NP] = I;
      }
    }
    if (NN > -1 && NP < M) {
      for (int S = 0; S < M - 1; S++) {
        int I = W[S];
        int J = W[NP];
        A[I] = J;
        Q[J] += Q[I] - 1.0;
        if (Q[J] < 1.0) {
          NP++;
        }
        if (NP >= M) {
          break;
        }
      }
      // A[W[M]] = W[M];
    }

    for (int I = 0; I < M; I++) {
      Q[I] += I;
    }

    // Do we need to keep track of how many copies to use?
    int[] counts = new int[M];

    int count = 0;
    for (int i = 0; i < m_Weights.length; i++) {
      int ALRV;
      double U = M * random.nextDouble();
      int I = (int) U;
      if (U < Q[I]) {
        ALRV = I;
      } else {
        ALRV = A[I];
      }
      counts[ALRV]++;
      if (!sampled[ALRV]) {
        sampled[ALRV] = true;
        count++;
      }
    }

    // Generate output
    double[][] output = new double[2][count];
    int index = 0;
    for (int i = 0; i < M; i++) {
      if (counts[i] > 0) {
        output[0][index] = m_Values[i];
        output[1][index] = counts[i];
        index++;
      }
    }

    return output;
  }

  /**
   * Selects the number of components using leave-one-out Bootstrap, estimating loglikelihood.
   *
   * @return the number of components to use
   */
  protected int findNumComponentsUsingBootStrap() {

    if (m_NumComponents > 0) {
      return m_NumComponents;
    }
    if (m_MaxNumComponents <= 1) {
      return 1;
    }

    double bestLogLikelihood = -Double.MAX_VALUE;
    int bestNumComponents = 1;  
    for (int i = 1; i <= m_MaxNumComponents; i++) {
      double logLikelihood = 0;
      for (int k = 0; k < m_NumBootstrapRuns; k++) {
        boolean[] inBag = new boolean[m_NumValues];
        double[][] output = resampleWithWeights(m_Random, inBag);
        MM mixtureModel = buildModel(i, output[0], output[1]);
        double locLogLikelihood = 0;
        double totalWeight = 0;
        for (int j = 0; j < m_NumValues; j++) {
          if (!inBag[j]) {
            double weight = m_Weights[j];
            locLogLikelihood += weight * mixtureModel.logDensity(m_Values[j]);
            totalWeight += weight;
          }
        }
        locLogLikelihood /= totalWeight;
        logLikelihood += locLogLikelihood;
      }
      logLikelihood /= (double)m_NumBootstrapRuns;
      if (m_Debug) {
        System.err.println("Loglikelihood: " + logLikelihood + "\tNumber of components: " + i);
      }
      if (logLikelihood > bestLogLikelihood) {
        bestNumComponents = i;
        bestLogLikelihood = logLikelihood;
      }
    }

    return bestNumComponents;
  }

  /**
   * Calculates entrpy for given model and data.
   */
  protected double entropy(MM mixtureModel) {

    double entropy = 0;
    for (int j = 0; j < m_NumValues; j++) {
      entropy += m_Weights[j] * 
        ContingencyTables.entropy(Utils.logs2probs(mixtureModel.logJointDensities(m_Values[j])));
    }
    entropy *= Utils.log2; // Need natural logarithm, not base-2 logarithm

    return entropy / (double)m_NumValues;
  }

  /**
   * Selects the number of components using normalized entropy.
   *
   * @return the model to use
   */
  protected MM findModelUsingNormalizedEntropy() {

    if (m_NumComponents > 0) {
      return buildModel(m_NumComponents, m_Values, m_Weights);
    }
    if (m_MaxNumComponents <= 1) {
      return buildModel(1, m_Values, m_Weights);
    }

    //Loglikelihood for one cluster
    MM bestMixtureModel = buildModel(1, m_Values, m_Weights);
    double loglikelihoodForOneCluster = bestMixtureModel.loglikelihood(m_Values, m_Weights);
    double bestNormalizedEntropy = 1;
    for (int i = 2; i <= m_MaxNumComponents; i++) {
      MM mixtureModel = buildModel(i, m_Values, m_Weights);

      double loglikelihood = mixtureModel.loglikelihood(m_Values, m_Weights);
      if (loglikelihood < loglikelihoodForOneCluster) {
        // This appears to happen in practice, hopefully not because of a bug...
        if (m_Debug) {
          System.err.println("Likelihood for one cluster greater than for " + i + " clusters.");
        }
        continue;
      }     
      double entropy = entropy(mixtureModel);
      double normalizedEntropy = entropy / (loglikelihood - loglikelihoodForOneCluster);

      if (m_Debug) {
        System.err.println("Entropy: " + entropy + "\tLogLikelihood: " + loglikelihood +
          "\tLoglikelihood for one cluster: " + loglikelihoodForOneCluster + "\tNormalized entropy: " + normalizedEntropy + "\tNumber of components: " + i);
      }
      if (normalizedEntropy < bestNormalizedEntropy) {
        bestMixtureModel = mixtureModel;
        bestNormalizedEntropy = normalizedEntropy;
      }
    }

    return bestMixtureModel;
  }

  /**
   * Updates the model based on the current data.
   * Uses the leave-one-out Bootstrap to choose the number of components.
   */
  protected void updateModel() {

    if (m_MixtureModel != null) {
      return;
    } else if (m_NumValues > 0) {

      // Shrink arrays if necessary
      if (m_Values.length > m_NumValues) {
        double[] values = new double[m_NumValues];
        double[] weights = new double[m_NumValues];
        System.arraycopy(m_Values,  0,  values,  0,  m_NumValues);
        System.arraycopy(m_Weights,  0,  weights,  0,  m_NumValues);
        m_Values = values;
        m_Weights = weights;
      }

      if (m_UseNormalizedEntropy) {
        m_MixtureModel = findModelUsingNormalizedEntropy();
      } else {
        m_MixtureModel = buildModel(findNumComponentsUsingBootStrap(), m_Values, m_Weights);
      }
    }
  }

  /**
   * Returns the interval for the given confidence value. 
   * 
   * @param conf the confidence value in the interval [0, 1]
   * @return the interval
   */
  public double[][] predictIntervals(double conf) {

    updateModel();

    return m_MixtureModel.predictIntervals(conf);
  }

  /**
   * Returns the quantile for the given percentage.
   * 
   * @param percentage the percentage
   * @return the quantile
   */
  public double predictQuantile(double percentage) {

    updateModel();

    return m_MixtureModel.predictQuantile(percentage);
  }


  /**
   * Returns the natural logarithm of the density estimate at the given
   * point.
   *
   * @param value the value at which to evaluate
   * @return the natural logarithm of the density estimate at the given
   * value
   */
  public double logDensity(double value) {

    updateModel();
    if (m_MixtureModel == null) {
      return Math.log(Double.MIN_VALUE);
    }
    return m_MixtureModel.logDensity(value);
  }

  /**
   * Returns textual description of this estimator.
   */
  public String toString() {

    updateModel();
    if (m_MixtureModel == null) {
      return "";
    }
    return m_MixtureModel.toString();
  }

  /**
   * Returns an enumeration that lists the command-line options that are available
   * 
   * @return the list of options as an enumeration
   */
  @Override
  public Enumeration




© 2015 - 2025 Weber Informatics LLC | Privacy Policy