Struct WeightedIndex
struct WeightedIndex<X: SampleUniform + PartialOrd> { ... }
A distribution using weighted sampling of discrete items.
Sampling a WeightedIndex distribution returns the index of a randomly
selected element from the iterator used when the WeightedIndex was
created. The chance of a given element being picked is proportional to the
weight of the element. The weights can use any type X for which an
implementation of Uniform<X> exists. The implementation guarantees that
elements with zero weight are never picked, even when the weights are
floating point numbers.
Performance
Time complexity of sampling from WeightedIndex is O(log N) where
N is the number of weights.
See also rand_distr::weighted for alternative implementations supporting
potentially-faster sampling or a more easily modifiable tree structure.
A WeightedIndex<X> contains a Vec<X> and a Uniform<X> and so its
size is the sum of the size of those objects, possibly plus some alignment.
Creating a WeightedIndex<X> will allocate enough space to hold N - 1
weights of type X, where N is the number of weights. However, since
Vec doesn't guarantee a particular growth strategy, additional memory
might be allocated but not used. Since the WeightedIndex object also
contains an instance of X::Sampler, this might cause additional allocations,
though for primitive types, Uniform<X> doesn't allocate any memory.
Sampling from WeightedIndex will result in a single call to
Uniform<X>::sample (method of the Distribution trait), which typically
will request a single value from the underlying RngCore, though the
exact number depends on the implementation of Uniform<X>::sample.
Example
use *;
use WeightedIndex;
let choices = ;
let weights = ;
let dist = new.unwrap;
let mut rng = rng;
for _ in 0..100
let items = ;
let dist2 = new.unwrap;
for _ in 0..100
Implementations
impl<X: SampleUniform + PartialOrd + Clone> WeightedIndex<X>
fn weight(self: &Self, index: usize) -> Option<X> where X: for<'a> SubAssign<&'a X>Returns the weight at the given index, if it exists.
If the index is out of bounds, this will return
None.Example
use WeightedIndex; let weights = ; let dist = new.unwrap; assert_eq!; assert_eq!; assert_eq!; assert_eq!;fn weights(self: &Self) -> WeightedIndexIter<'_, X> where X: for<'a> SubAssign<&'a X>Returns a lazy-loading iterator containing the current weights of this distribution.
If this distribution has not been updated since its creation, this will return the same weights as were passed to
new.Example
use WeightedIndex; let weights = ; let mut dist = new.unwrap; assert_eq!; dist.update_weights.unwrap; assert_eq!;fn total_weight(self: &Self) -> XReturns the sum of all weights in this distribution.
impl<X: SampleUniform + PartialOrd> WeightedIndex<X>
fn new<I>(weights: I) -> Result<WeightedIndex<X>, Error> where I: IntoIterator, <I as >::Item: SampleBorrow<X>, X: WeightCreates a new a
WeightedIndexDistributionusing the values inweights. The weights can use any typeXfor which an implementation ofUniform<X>exists.Error cases:
Error::InvalidInputwhen the iteratorweightsis empty.Error::InvalidWeightwhen a weight is not-a-number or negative.Error::InsufficientNonZerowhen the sum of all weights is zero.Error::Overflowwhen the sum of all weights overflows.
fn update_weights(self: &mut Self, new_weights: &[(usize, &X)]) -> Result<(), Error> where X: for<'a> AddAssign<&'a X> + for<'a> SubAssign<&'a X> + Clone + DefaultUpdate a subset of weights, without changing the number of weights.
new_weightsmust be sorted by the index.Using this method instead of
newmight be more efficient if only a small number of weights is modified. No allocations are performed, unless the weight typeXuses allocation internally.In case of error,
selfis not modified. Error cases:Error::InvalidInputwhennew_weightsare not ordered by index or an index is too large.Error::InvalidWeightwhen a weight is not-a-number or negative.Error::InsufficientNonZerowhen the sum of all weights is zero. Note that due to floating-point loss of precision, this case is not always correctly detected; usage of a fixed-point weight type may be preferred.
Updates take
O(N)time. If you need to frequently update weights, considerrand_distr::weighted_treeas an alternative where an update isO(log N).
impl<'de, X> Deserialize for WeightedIndex<X>
fn deserialize<__D>(__deserializer: __D) -> Result<Self, <__D as >::Error> where __D: Deserializer<'de>
impl<T> Any for WeightedIndex<X>
fn type_id(self: &Self) -> TypeId
impl<T> Borrow for WeightedIndex<X>
fn borrow(self: &Self) -> &T
impl<T> BorrowMut for WeightedIndex<X>
fn borrow_mut(self: &mut Self) -> &mut T
impl<T> CloneToUninit for WeightedIndex<X>
unsafe fn clone_to_uninit(self: &Self, dest: *mut u8)
impl<T> DeserializeOwned for WeightedIndex<X>
impl<T> From for WeightedIndex<X>
fn from(t: T) -> TReturns the argument unchanged.
impl<T> ToOwned for WeightedIndex<X>
fn to_owned(self: &Self) -> Tfn clone_into(self: &Self, target: &mut T)
impl<T, U> Into for WeightedIndex<X>
fn into(self: Self) -> UCalls
U::from(self).That is, this conversion is whatever the implementation of
[From]<T> for Uchooses to do.
impl<T, U> TryFrom for WeightedIndex<X>
fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>
impl<T, U> TryInto for WeightedIndex<X>
fn try_into(self: Self) -> Result<U, <U as TryFrom<T>>::Error>
impl<V, T> VZip for WeightedIndex<X>
fn vzip(self: Self) -> V
impl<X> Distribution for WeightedIndex<X>
fn sample<R: Rng + ?Sized>(self: &Self, rng: &mut R) -> usize
impl<X> Freeze for WeightedIndex<X>
impl<X> RefUnwindSafe for WeightedIndex<X>
impl<X> Send for WeightedIndex<X>
impl<X> Serialize for WeightedIndex<X>
fn serialize<__S>(self: &Self, __serializer: __S) -> Result<<__S as >::Ok, <__S as >::Error> where __S: Serializer
impl<X> Sync for WeightedIndex<X>
impl<X> Unpin for WeightedIndex<X>
impl<X> UnsafeUnpin for WeightedIndex<X>
impl<X> UnwindSafe for WeightedIndex<X>
impl<X: $crate::clone::Clone + SampleUniform + PartialOrd> Clone for WeightedIndex<X>
fn clone(self: &Self) -> WeightedIndex<X>
impl<X: $crate::cmp::PartialEq + SampleUniform + PartialOrd> PartialEq for WeightedIndex<X>
fn eq(self: &Self, other: &WeightedIndex<X>) -> bool
impl<X: $crate::fmt::Debug + SampleUniform + PartialOrd> Debug for WeightedIndex<X>
fn fmt(self: &Self, f: &mut Formatter<'_>) -> Result