pumpkin_solver/branching/value_selection/
in_domain_interval.rsuse super::InDomainSplit;
use crate::branching::SelectionContext;
use crate::branching::ValueSelector;
use crate::engine::predicates::predicate::Predicate;
use crate::engine::variables::DomainId;
use crate::predicate;
#[derive(Debug, Copy, Clone)]
pub struct InDomainInterval;
impl ValueSelector<DomainId> for InDomainInterval {
fn select_value(
&mut self,
context: &mut SelectionContext,
decision_variable: DomainId,
) -> Predicate {
if let Some(first_interval) = (context.lower_bound(decision_variable) + 1
..context.upper_bound(decision_variable))
.find(|bound| !context.contains(decision_variable, *bound))
{
predicate!(decision_variable <= first_interval - 1)
} else {
InDomainSplit::get_predicate_excluding_upper_half(context, decision_variable)
}
}
}
#[cfg(test)]
mod tests {
use super::InDomainInterval;
use crate::basic_types::tests::TestRandom;
use crate::branching::SelectionContext;
use crate::branching::ValueSelector;
use crate::predicate;
#[test]
fn test_returns_correct_literal() {
let (mut assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(1, 0, Some(vec![(0, 10)]));
let mut test_rng = TestRandom::default();
let domain_ids = assignments_integer.get_domains().collect::<Vec<_>>();
let mut selector = InDomainInterval;
for to_remove in [2, 3, 7, 8] {
let _ = assignments_integer.remove_value_from_domain(domain_ids[0], to_remove, None);
}
let mut context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let selected_predicate = selector.select_value(&mut context, domain_ids[0]);
assert_eq!(selected_predicate, predicate!(domain_ids[0] <= 1))
}
#[test]
fn test_no_holes_in_domain_bisects_domain() {
let (assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(1, 0, Some(vec![(0, 10)]));
let mut test_rng = TestRandom::default();
let mut context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let domain_ids = context.get_domains().collect::<Vec<_>>();
let mut selector = InDomainInterval;
let selected_predicate = selector.select_value(&mut context, domain_ids[0]);
assert_eq!(selected_predicate, predicate!(domain_ids[0] <= 5),)
}
#[test]
fn test_domain_of_size_two() {
let (assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(1, 0, Some(vec![(1, 2)]));
let mut test_rng = TestRandom::default();
let mut context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let domain_ids = context.get_domains().collect::<Vec<_>>();
let mut selector = InDomainInterval;
let selected_predicate = selector.select_value(&mut context, domain_ids[0]);
assert_eq!(selected_predicate, predicate!(domain_ids[0] <= 1))
}
}