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.
 
 
 
 
 
 

278 lines
6.2 KiB

//==============================================================================
//
// Copyright (c) 2002-
// Authors:
// * Dave Parker <david.parker@comlab.ox.ac.uk> (University of Oxford)
//
//------------------------------------------------------------------------------
//
// This file is part of PRISM.
//
// PRISM 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 2 of the License, or
// (at your option) any later version.
//
// PRISM 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 PRISM; if not, write to the Free Software Foundation,
// Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
//
//==============================================================================
package explicit;
import java.util.BitSet;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import prism.PrismUtils;
/**
* Explicit representation of a probability distribution.
* Basically, a mapping from (integer-valued) indices to (non-zero, double-valued) probabilities.
*/
public class Distribution implements Iterable<Entry<Integer, Double>>
{
private HashMap<Integer, Double> map;
/**
* Create an empty distribution.
*/
public Distribution()
{
clear();
}
/**
* Copy constructor.
*/
public Distribution(Distribution distr)
{
this(distr.iterator());
}
/**
* Construct a distribution from an iterator over transitions.
*/
public Distribution(Iterator<Entry<Integer, Double>> transitions)
{
this();
while (transitions.hasNext()) {
final Entry<Integer, Double> trans = transitions.next();
add(trans.getKey(), trans.getValue());
}
}
/**
* Construct a distribution from an existing one and an index permutation,
* i.e. in which index i becomes index permut[i].
* Note: have to build the new distributions from scratch anyway to do this,
* so may as well provide this functionality as a constructor.
*/
public Distribution(Distribution distr, int permut[])
{
this();
Iterator<Entry<Integer, Double>> i = distr.iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
add(permut[e.getKey()], e.getValue());
}
}
/**
* Clear all entries of the distribution.
*/
public void clear()
{
map = new HashMap<Integer, Double>();
}
/**
* Add 'prob' to the probability for index 'j'.
* Return boolean indicating whether or not there was already
* non-zero probability for this index (i.e. false denotes new transition).
*/
public boolean add(int j, double prob)
{
Double d = (Double) map.get(j);
if (d == null) {
map.put(j, prob);
return false;
} else {
set(j, d + prob);
return true;
}
}
/**
* Set the probability for index 'j' to 'prob'.
*/
public void set(int j, double prob)
{
if (prob == 0.0)
map.remove(j);
else
map.put(j, prob);
}
/**
* Get the probability for index j.
*/
public double get(int j)
{
Double d;
d = (Double) map.get(j);
return d == null ? 0.0 : d.doubleValue();
}
/**
* Returns true if index j is in the support of the distribution.
*/
public boolean contains(int j)
{
return map.get(j) != null;
}
/**
* Returns true if all indices in the support of the distribution are in the set.
*/
public boolean isSubsetOf(BitSet set)
{
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
if (!set.get((Integer) e.getKey()))
return false;
}
return true;
}
/**
* Returns true if at least one index in the support of the distribution is in the set.
*/
public boolean containsOneOf(BitSet set)
{
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
if (set.get((Integer) e.getKey()))
return true;
}
return false;
}
/**
* Get the support of the distribution.
*/
public Set<Integer> getSupport()
{
return map.keySet();
}
/**
* Get an iterator over the entries of the map defining the distribution.
*/
public Iterator<Entry<Integer, Double>> iterator()
{
return map.entrySet().iterator();
}
/**
* Returns true if the distribution is empty.
*/
public boolean isEmpty()
{
return map.isEmpty();
}
/**
* Get the size of the support of the distribution.
*/
public int size()
{
return map.size();
}
/**
* Get the sum of the probabilities in the distribution.
*/
public double sum()
{
double d = 0.0;
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
d += e.getValue();
}
return d;
}
/**
* Get the sum of all the probabilities in the distribution except for index j.
*/
public double sumAllBut(int j)
{
double d = 0.0;
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
if (e.getKey() != j)
d += e.getValue();
}
return d;
}
/**
* Create a new distribution, based on a mapping from the indices
* used in this distribution to a different set of indices.
*/
public Distribution map(int map[])
{
Distribution distrNew = new Distribution();
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
distrNew.add(map[e.getKey()], e.getValue());
}
return distrNew;
}
@Override
public boolean equals(Object o)
{
Double d1, d2;
Distribution d = (Distribution) o;
if (d.size() != size())
return false;
Iterator<Entry<Integer, Double>> i = iterator();
while (i.hasNext()) {
Map.Entry<Integer, Double> e = i.next();
d1 = e.getValue();
d2 = d.map.get(e.getKey());
if (d2 == null || !PrismUtils.doublesAreClose(d1, d2, 1e-12, false))
return false;
}
return true;
}
@Override
public int hashCode()
{
// Simple hash code
return map.size();
}
@Override
public String toString()
{
return map.toString();
}
}