pumpkin_solver/branching/variable_selection/
occurrence.rsuse log::warn;
use crate::branching::Direction;
use crate::branching::InOrderTieBreaker;
use crate::branching::SelectionContext;
use crate::branching::TieBreaker;
use crate::branching::VariableSelector;
use crate::engine::variables::DomainId;
use crate::pumpkin_assert_eq_simple;
pub struct Occurrence<Var, TieBreaking> {
variables: Vec<Var>,
tie_breaker: TieBreaking,
num_occurrences: Vec<u32>,
}
impl<Var, TieBreaking> std::fmt::Debug for Occurrence<Var, TieBreaking> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Occurrence").finish()
}
}
impl<Var: Copy> Occurrence<Var, InOrderTieBreaker<Var, u32>> {
pub fn new(variables: &[Var], num_occurrences: &[u32]) -> Self {
pumpkin_assert_eq_simple!(
variables.len(), num_occurrences.len(),
"The number of variables and the number of elements in num_occurrences for the Occurence variable selector should be the same"
);
if variables.is_empty() {
warn!("The Occurence variable selector was not provided with any variables");
}
Occurrence {
variables: variables.to_vec(),
tie_breaker: InOrderTieBreaker::new(Direction::Maximum),
num_occurrences: num_occurrences.to_vec(),
}
}
}
impl<TieBreaking> VariableSelector<DomainId> for Occurrence<DomainId, TieBreaking>
where
TieBreaking: TieBreaker<DomainId, u32>,
{
fn select_variable(&mut self, context: &SelectionContext) -> Option<DomainId> {
self.variables
.iter()
.enumerate()
.filter(|(_, variable)| !context.is_integer_fixed(**variable))
.for_each(|(index, variable)| {
self.tie_breaker
.consider(*variable, self.num_occurrences[index])
});
self.tie_breaker.select()
}
}
#[cfg(test)]
mod tests {
use crate::basic_types::tests::TestRandom;
use crate::branching::Occurrence;
use crate::branching::SelectionContext;
use crate::branching::VariableSelector;
#[test]
fn test_correctly_selected() {
let (assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(2, 0, Some(vec![(0, 10), (10, 20)]));
let mut test_rng = TestRandom::default();
let context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let integer_variables = context.get_domains().collect::<Vec<_>>();
let mut strategy = Occurrence::new(&integer_variables, &[2, 1]);
let selected = strategy.select_variable(&context);
assert!(selected.is_some());
assert_eq!(selected.unwrap(), integer_variables[0])
}
#[test]
fn fixed_variables_are_not_selected() {
let (assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(2, 0, Some(vec![(10, 10), (20, 20)]));
let mut test_rng = TestRandom::default();
let context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let integer_variables = context.get_domains().collect::<Vec<_>>();
let mut strategy = Occurrence::new(&integer_variables, &[1, 2]);
let selected = strategy.select_variable(&context);
assert!(selected.is_none());
}
}