/* Copyright 2006 by Sean Luke Licensed under the Academic Free License version 3.0 See the file "LICENSE" for more information */ package ec.util; /* * RandomChoice.java * * Created: January 5, 2001 * By: Sean Luke * */ /** * RandomChoice organizes arrays of floats into distributions which can * be used to pick randomly from. You can provide three kinds of arrays: * * *

Before the RandomChoice can pick randomly from your array, it must first * organize it. It does this by doing the following. First, it normalizes the * values in the array. Then it modifies them to their sums. That is, each item i * in the array is set to the sum of the original values for items 0...i. If you * cannot allow your objects to be modified, then this is not the class for you. * *

An array is valid if (1) it has no negative values and (2) not all of its * values are zero. This RandomChoice code should (I hope) guarantee that * an element of zero probability is never returned. RandomChoice uses a binary * search to find your index, followed by linear probing (marching up or down * the list) to find the first non-zero probability item in the vacinity of that * index. As long as there are not a whole lot of zero-valued items in a row, * RandomChoice is efficient. * * You organize your array with organizeDistribution(). Then you can have the * RandomChoice pick random items from the array and return their indexes to you. * You do this by calling pickFromDistribution(), passing it a random floating * point value between 0.0 and 1.0. You call organizeDistribution() only once; * after which you may call pickFromDistribution() as many times as you like. * You should not modify the array thereafter. * * @author Sean Luke * @version 1.0 */ public class RandomChoice { /** Same as organizeDistribution(probabilities, false); */ public static void organizeDistribution(final float[] probabilities) { organizeDistribution(probabilities, false); } /** Normalizes probabilities, then converts them into continuing sums. This prepares them for being usable in pickFromDistribution. If the probabilities are all 0, then selection is uniform, unless allowAllZeros is false, in which case an ArithmeticException is thrown. If any of them are negative, or if the distribution is empty, then an ArithmeticException is thrown. For example, {0.6, 0.4, 0.2, 0.8} -> {0.3, 0.2, 0.1, 0.4} -> {0.3, 0.5, 0.6, 1.0} */ public static void organizeDistribution(final float[] probabilities, final boolean allowAllZeros) { // first normalize double sum=0.0; if (probabilities.length == 0) throw new ArithmeticException("Distribution has no elements"); for(int x=0;x 0; x--) if (probabilities[x]==probabilities[x-1]) // we're 0.0 probabilities[x] = 1.0f; else break; probabilities[x] = 1.0f; } /** Same as organizeDistribution(probabilities, false); */ public static void organizeDistribution(final double[] probabilities) { organizeDistribution(probabilities, false); } /** Normalizes probabilities, then converts them into continuing sums. This prepares them for being usable in pickFromDistribution. If the probabilities are all 0, then selection is uniform, unless allowAllZeros is false, in which case an ArithmeticException is thrown. If any of them are negative, or if the distribution is empty, then an ArithmeticException is thrown. For example, {0.6, 0.4, 0.2, 0.8} -> {0.3, 0.2, 0.1, 0.4} -> {0.3, 0.5, 0.6, 1.0} */ public static void organizeDistribution(final double[] probabilities, final boolean allowAllZeros) { // first normalize double sum=0.0; if (probabilities.length == 0) throw new ArithmeticException("Distribution has no elements"); for(int x=0;x 0; x--) if (probabilities[x]==probabilities[x-1]) // we're 0.0 probabilities[x] = 1.0; else break; probabilities[x] = 1.0; } /** Same as organizeDistribution(objs, chooser, false); */ public static void organizeDistribution(final Object[] objs, final RandomChoiceChooser chooser) { organizeDistribution(objs,chooser, false); } /** Normalizes the probabilities associated with an array of objects, then converts them into continuing sums. This prepares them for being usable in pickFromDistribution. If the probabilities are all 0, then selection is uniform, unless allowAllZeros is false, in which case an ArithmeticException is thrown. If any of them are negative, or if the distribution is empty, then an ArithmeticException is thrown. For example, {0.6, 0.4, 0.2, 0.8} -> {0.3, 0.2, 0.1, 0.4} -> {0.3, 0.5, 0.6, 1.0} The probabilities are retrieved and set using chooser.*/ public static void organizeDistribution(final Object[] objs, final RandomChoiceChooser chooser, final boolean allowAllZeros) { // first normalize double sum=0.0; if (objs.length == 0) throw new ArithmeticException("Distribution has no elements"); for(int x=0;x 0; x--) if (chooser.getProbability(objs[x])== chooser.getProbability(objs[x-1])) // we're 0.0 chooser.setProbability(objs[x],1.0f); else break; chooser.setProbability(objs[x],1.0f); } /** Same as organizeDistribution(objs, chooser, false); */ public static void organizeDistribution(final Object[] objs, final RandomChoiceChooserD chooser) { organizeDistribution(objs,chooser, false); } /** Normalizes the probabilities associated with an array of objects, then converts them into continuing sums. This prepares them for being usable in pickFromDistribution. If the probabilities are all 0, then selection is uniform, unless allowAllZeros is false, in which case an ArithmeticException is thrown. If any of them are negative, or if the distribution is empty, then an ArithmeticException is thrown. For example, {0.6, 0.4, 0.2, 0.8} -> {0.3, 0.2, 0.1, 0.4} -> {0.3, 0.5, 0.6, 1.0} The probabilities are retrieved and set using chooser.*/ public static void organizeDistribution(final Object[] objs, final RandomChoiceChooserD chooser, final boolean allowAllZeros) { // first normalize double sum=0.0; if (objs.length == 0) throw new ArithmeticException("Distribution has no elements"); for(int x=0;x 0; x--) if (chooser.getProbability(objs[x])== chooser.getProbability(objs[x-1])) // we're 0.0 chooser.setProbability(objs[x],1.0); else break; chooser.setProbability(objs[x],1.0); } // allows us to have zero-probability values private static final int exemptZeroes(final float[] probabilities, int index) { //System.out.println(index); if (probabilities[index]==0.0f) // I need to scan forward because I'm in a left-trail // scan forward { while(index < probabilities.length-1 && probabilities[index]==0.0f) index++; } else // scan backwards { while(index > 0 && probabilities[index]==probabilities[index-1]) index--; } return index; } // allows us to have zero-probability values private static final int exemptZeroes(final double[] probabilities, int index) { //System.out.println(index); if (probabilities[index]==0.0) // I need to scan forward because I'm in a left-trail // scan forward { while(index < probabilities.length-1 && probabilities[index]==0.0) index++; } else // scan backwards { while(index > 0 && probabilities[index]==probabilities[index-1]) index--; } return index; } // allows us to have zero-probability values private static final int exemptZeroes(final Object[] objs, final RandomChoiceChooser chooser, int index) { //System.out.println(index); if (chooser.getProbability(objs[index])==0.0f) // I need to scan forward because I'm in a left-trail // scan forward { while(index < objs.length-1 && chooser.getProbability(objs[index])==0.0f) index++; } else // scan backwards { while(index > 0 && chooser.getProbability(objs[index])== chooser.getProbability(objs[index-1])) index--; } return index; } // allows us to have zero-probability values private static final int exemptZeroes(final Object[] objs, final RandomChoiceChooserD chooser, int index) { //System.out.println(index); if (chooser.getProbability(objs[index])==0.0) // I need to scan forward because I'm in a left-trail // scan forward { while(index < objs.length-1 && chooser.getProbability(objs[index])==0.0) index++; } else // scan backwards { while(index > 0 && chooser.getProbability(objs[index])== chooser.getProbability(objs[index-1])) index--; } return index; } public static final int CHECKBOUNDARY = 8; /** Picks a random item from an array of probabilities, normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < CHECKBOUNDARY, then a linear search is used, else a binary search is used. */ public static int pickFromDistribution(final float[] probabilities, final float prob) { return pickFromDistribution(probabilities, prob, CHECKBOUNDARY); } /** Picks a random item from an array of probabilities, normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < checkboundary, then a linear search is used, else a binary search is used. @deprecated */ public static int pickFromDistribution(final float[] probabilities, final float prob, final int checkboundary) { if (prob<0.0f || prob>1.0f) throw new ArithmeticException("Invalid probability for pickFromDistribution (must be 0.0<=x<=1.0)"); else if (probabilities.length==1) // quick return 0; else if (probabilities.lengthprob) return exemptZeroes(probabilities,x); return exemptZeroes(probabilities,probabilities.length-1); } else { // binary search int top = probabilities.length-1; int bottom = 0; int cur; while(top!=bottom) { cur = (top + bottom) / 2; // integer division if (probabilities[cur] > prob) if (cur==0 || probabilities[cur-1] <= prob) return exemptZeroes(probabilities,cur); else // step down top = cur; else if (cur==probabilities.length-1) // oops return exemptZeroes(probabilities,cur); else if (bottom==cur) // step up bottom++; // (8 + 9)/2 = 8 else bottom = cur; // (8 + 10) / 2 = 9 } return exemptZeroes(probabilities,bottom); // oops } } /** Picks a random item from an array of probabilities, normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < CHECKBOUNDARY, then a linear search is used, else a binary search is used. */ public static int pickFromDistribution(final double[] probabilities, final double prob) { return pickFromDistribution(probabilities, prob, CHECKBOUNDARY); } /** Picks a random item from an array of probabilities, normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < checkboundary, then a linear search is used, else a binary search is used. @deprecated */ public static int pickFromDistribution(final double[] probabilities, final double prob, final int checkboundary) { if (prob<0.0 || prob>1.0) throw new ArithmeticException("Invalid probability for pickFromDistribution (must be 0.0<=x<=1.0)"); if (probabilities.length==1) // quick return 0; else if (probabilities.lengthprob) return exemptZeroes(probabilities,x); return exemptZeroes(probabilities,probabilities.length-1); } else { // binary search int top = probabilities.length-1; int bottom = 0; int cur; while(top!=bottom) { cur = (top + bottom) / 2; // integer division if (probabilities[cur] > prob) if (cur==0 || probabilities[cur-1] <= prob) return exemptZeroes(probabilities,cur); else // step down top = cur; else if (cur==probabilities.length-1) // oops return exemptZeroes(probabilities,cur); else if (bottom==cur) // step up bottom++; // (8 + 9)/2 = 8 else bottom = cur; // (8 + 10) / 2 = 9 } return exemptZeroes(probabilities,bottom); // oops } } /** Picks a random item from an array of objects, each with an associated probability that is accessed by taking an object and passing it to chooser.getProbability(obj). The objects' probabilities are normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < CHECKBOUNDARY, then a linear search is used, else a binary search is used. */ public static int pickFromDistribution(final Object[] objs, final RandomChoiceChooser chooser, final float prob) { return pickFromDistribution(objs, chooser, prob, CHECKBOUNDARY); } /** Picks a random item from an array of objects, each with an associated probability that is accessed by taking an object and passing it to chooser.getProbability(obj). The objects' probabilities are normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < checkboundary, then a linear search is used, else a binary search is used. @deprecated */ public static int pickFromDistribution(final Object[] objs, final RandomChoiceChooser chooser, final float prob, final int checkboundary) { if (prob<0.0f || prob>1.0f) throw new ArithmeticException("Invalid probability for pickFromDistribution (must be 0.0<=x<=1.0)"); if (objs.length==1) // quick return 0; else if (objs.lengthprob) return exemptZeroes(objs,chooser,x); return exemptZeroes(objs,chooser,objs.length-1); } else { // binary search int top = objs.length-1; int bottom = 0; int cur; while(top!=bottom) { cur = (top + bottom) / 2; // integer division if (chooser.getProbability(objs[cur]) > prob) if (cur==0 || chooser.getProbability(objs[cur-1]) <= prob) return exemptZeroes(objs,chooser,cur); else // step down top = cur; else if (cur==objs.length-1) // oops return exemptZeroes(objs,chooser,cur); else if (bottom==cur) // step up bottom++; // (8 + 9)/2 = 8 else bottom = cur; // (8 + 10) / 2 = 9 } return exemptZeroes(objs,chooser,bottom); // oops } } /** Picks a random item from an array of objects, each with an associated probability that is accessed by taking an object and passing it to chooser.getProbability(obj). The objects' probabilities are normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < CHECKBOUNDARY, then a linear search is used, else a binary search is used. */ public static int pickFromDistribution(final Object[] objs, final RandomChoiceChooserD chooser, final double prob) { return pickFromDistribution(objs, chooser, prob, CHECKBOUNDARY); } /** Picks a random item from an array of objects, each with an associated probability that is accessed by taking an object and passing it to chooser.getProbability(obj). The objects' probabilities are normalized and summed as follows: For example, if four probabilities are {0.3, 0.2, 0.1, 0.4}, then they should get normalized and summed by the outside owners as: {0.3, 0.5, 0.6, 1.0}. If probabilities.length < checkboundary, then a linear search is used, else a binary search is used. @deprecated */ public static int pickFromDistribution(final Object[] objs, final RandomChoiceChooserD chooser, final double prob, final int checkboundary) { if (prob<0.0 || prob>1.0) throw new ArithmeticException("Invalid probability for pickFromDistribution (must be 0.0<=x<=1.0)"); if (objs.length==1) // quick return 0; else if (objs.lengthprob) return exemptZeroes(objs,chooser,x); return exemptZeroes(objs,chooser,objs.length-1); } else { // binary search int top = objs.length-1; int bottom = 0; int cur; while(top!=bottom) { cur = (top + bottom) / 2; // integer division if (chooser.getProbability(objs[cur]) > prob) if (cur==0 || chooser.getProbability(objs[cur-1]) <= prob) return exemptZeroes(objs,chooser,cur); else // step down top = cur; else if (cur==objs.length-1) // oops return exemptZeroes(objs,chooser,cur); else if (bottom==cur) // step up bottom++; // (8 + 9)/2 = 8 else bottom = cur; // (8 + 10) / 2 = 9 } return exemptZeroes(objs,chooser,bottom); // oops } } }