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}