Skip to main content

solverforge_solver/heuristic/selector/
list_reverse.rs

1//! List reverse move selector for 2-opt optimization.
2//!
3//! Generates `ListReverseMove`s that reverse contiguous segments within a single
4//! list. This is the fundamental 2-opt move for TSP and VRP: reversing a segment
5//! of the tour can eliminate crossing edges and reduce total distance.
6//!
7//! For VRP, 2-opt is applied independently within each route (intra-route 2-opt).
8//! Cross-route 2-opt would require inter-entity reversal, which is a different
9//! operation modeled by `SubListSwapMove` with same-size segments.
10//!
11//! # Complexity
12//!
13//! For n entities with average route length m:
14//! O(n * m²) — all (start, end) pairs per entity where end > start + 1.
15//!
16//! # Example
17//!
18//! ```
19//! use solverforge_solver::heuristic::selector::list_reverse::ListReverseMoveSelector;
20//! use solverforge_solver::heuristic::selector::entity::FromSolutionEntitySelector;
21//! use solverforge_solver::heuristic::selector::MoveSelector;
22//! use solverforge_core::domain::PlanningSolution;
23//! use solverforge_core::score::SimpleScore;
24//!
25//! #[derive(Clone, Debug)]
26//! struct Vehicle { visits: Vec<i32> }
27//!
28//! #[derive(Clone, Debug)]
29//! struct Solution { vehicles: Vec<Vehicle>, score: Option<SimpleScore> }
30//!
31//! impl PlanningSolution for Solution {
32//!     type Score = SimpleScore;
33//!     fn score(&self) -> Option<Self::Score> { self.score }
34//!     fn set_score(&mut self, score: Option<Self::Score>) { self.score = score; }
35//! }
36//!
37//! fn list_len(s: &Solution, entity_idx: usize) -> usize {
38//!     s.vehicles.get(entity_idx).map_or(0, |v| v.visits.len())
39//! }
40//! fn list_reverse(s: &mut Solution, entity_idx: usize, start: usize, end: usize) {
41//!     if let Some(v) = s.vehicles.get_mut(entity_idx) {
42//!         v.visits[start..end].reverse();
43//!     }
44//! }
45//!
46//! let selector = ListReverseMoveSelector::<Solution, i32, _>::new(
47//!     FromSolutionEntitySelector::new(0),
48//!     list_len,
49//!     list_reverse,
50//!     "visits",
51//!     0,
52//! );
53//! ```
54
55use std::fmt::Debug;
56use std::marker::PhantomData;
57
58use solverforge_core::domain::PlanningSolution;
59use solverforge_scoring::ScoreDirector;
60
61use crate::heuristic::r#move::{ListMoveImpl, ListReverseMove};
62
63use super::entity::EntitySelector;
64use super::typed_move_selector::MoveSelector;
65
66/// A move selector that generates 2-opt segment reversal moves.
67///
68/// For each entity, enumerates all valid (start, end) pairs where
69/// `end > start + 1` (at least 2 elements in the reversed segment).
70///
71/// # Type Parameters
72/// * `S` - The solution type
73/// * `V` - The list element type (phantom — only used for type safety)
74/// * `ES` - The entity selector type
75pub struct ListReverseMoveSelector<S, V, ES> {
76    entity_selector: ES,
77    list_len: fn(&S, usize) -> usize,
78    list_reverse: fn(&mut S, usize, usize, usize),
79    variable_name: &'static str,
80    descriptor_index: usize,
81    _phantom: PhantomData<(fn() -> S, fn() -> V)>,
82}
83
84impl<S, V: Debug, ES: Debug> Debug for ListReverseMoveSelector<S, V, ES> {
85    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
86        f.debug_struct("ListReverseMoveSelector")
87            .field("entity_selector", &self.entity_selector)
88            .field("variable_name", &self.variable_name)
89            .field("descriptor_index", &self.descriptor_index)
90            .finish()
91    }
92}
93
94impl<S, V, ES> ListReverseMoveSelector<S, V, ES> {
95    /// Creates a new list reverse move selector.
96    ///
97    /// # Arguments
98    /// * `entity_selector` - Selects entities (routes) to apply 2-opt to
99    /// * `list_len` - Function to get route length
100    /// * `list_reverse` - Function to reverse `[start, end)` in-place
101    /// * `variable_name` - Name of the list variable
102    /// * `descriptor_index` - Entity descriptor index
103    pub fn new(
104        entity_selector: ES,
105        list_len: fn(&S, usize) -> usize,
106        list_reverse: fn(&mut S, usize, usize, usize),
107        variable_name: &'static str,
108        descriptor_index: usize,
109    ) -> Self {
110        Self {
111            entity_selector,
112            list_len,
113            list_reverse,
114            variable_name,
115            descriptor_index,
116            _phantom: PhantomData,
117        }
118    }
119}
120
121impl<S, V, ES> MoveSelector<S, ListReverseMove<S, V>> for ListReverseMoveSelector<S, V, ES>
122where
123    S: PlanningSolution,
124    V: Clone + Send + Sync + Debug + 'static,
125    ES: EntitySelector<S>,
126{
127    fn iter_moves<'a, D: ScoreDirector<S>>(
128        &'a self,
129        score_director: &'a D,
130    ) -> impl Iterator<Item = ListReverseMove<S, V>> + 'a {
131        let solution = score_director.working_solution();
132        let list_len = self.list_len;
133        let list_reverse = self.list_reverse;
134        let variable_name = self.variable_name;
135        let descriptor_index = self.descriptor_index;
136
137        let entities: Vec<usize> = self
138            .entity_selector
139            .iter(score_director)
140            .map(|r| r.entity_index)
141            .collect();
142
143        let mut moves = Vec::new();
144
145        for &entity in &entities {
146            let len = list_len(solution, entity);
147            if len < 2 {
148                continue;
149            }
150
151            // Enumerate all (start, end) pairs where end > start + 1
152            // This covers all 2-opt reversals within this entity's list
153            for start in 0..len {
154                // end is exclusive; minimum valid end = start + 2
155                for end in (start + 2)..=len {
156                    moves.push(ListReverseMove::new(
157                        entity,
158                        start,
159                        end,
160                        list_len,
161                        list_reverse,
162                        variable_name,
163                        descriptor_index,
164                    ));
165                }
166            }
167        }
168
169        moves.into_iter()
170    }
171
172    fn size<D: ScoreDirector<S>>(&self, score_director: &D) -> usize {
173        let solution = score_director.working_solution();
174        let list_len = self.list_len;
175
176        self.entity_selector
177            .iter(score_director)
178            .map(|r| {
179                let m = list_len(solution, r.entity_index);
180                // Number of valid (start, end) pairs: m*(m-1)/2 - m = m*(m-1)/2 - m
181                // For start in 0..m, end in start+2..=m: sum = m*(m-1)/2
182                if m >= 2 {
183                    m * (m - 1) / 2
184                } else {
185                    0
186                }
187            })
188            .sum()
189    }
190}
191
192/// Wraps a `ListReverseMoveSelector` to yield `ListMoveImpl::ListReverse`.
193pub struct ListMoveListReverseSelector<S, V, ES> {
194    inner: ListReverseMoveSelector<S, V, ES>,
195}
196
197impl<S, V: Debug, ES: Debug> Debug for ListMoveListReverseSelector<S, V, ES> {
198    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
199        f.debug_struct("ListMoveListReverseSelector")
200            .field("inner", &self.inner)
201            .finish()
202    }
203}
204
205impl<S, V, ES> ListMoveListReverseSelector<S, V, ES> {
206    /// Wraps an existing [`ListReverseMoveSelector`].
207    pub fn new(inner: ListReverseMoveSelector<S, V, ES>) -> Self {
208        Self { inner }
209    }
210}
211
212impl<S, V, ES> MoveSelector<S, ListMoveImpl<S, V>> for ListMoveListReverseSelector<S, V, ES>
213where
214    S: PlanningSolution,
215    V: Clone + PartialEq + Send + Sync + Debug + 'static,
216    ES: EntitySelector<S>,
217{
218    fn iter_moves<'a, D: ScoreDirector<S>>(
219        &'a self,
220        score_director: &'a D,
221    ) -> impl Iterator<Item = ListMoveImpl<S, V>> + 'a {
222        self.inner
223            .iter_moves(score_director)
224            .map(ListMoveImpl::ListReverse)
225    }
226
227    fn size<D: ScoreDirector<S>>(&self, score_director: &D) -> usize {
228        self.inner.size(score_director)
229    }
230}