// Use bitfields instead of HashSets. /* Or MAYBE, just use Vec ?! While that would use more memory, it would certainly be faster! Is saving a few bytes (these days) worth it, really? The iterator would be interesting... And we could handle any crazy size as well... :O */ use bit_field::BitField; // If I wanted to do 4x4 or 5x5, I would need more bits. (u32). use num::Integer; use std::ops::RangeBounds; use std::fmt; #[derive(Copy, Clone, PartialEq)] pub struct GenBits(pub T); impl GenBits { pub fn clear(&mut self) { self.0 = T::zero(); } pub fn set(&mut self, bit: usize, value: bool) { self.0.set_bit(bit, value); } #[must_use] pub fn get(&self, bit: usize) -> bool { self.0.get_bit(bit) } pub fn set_bits + IntoIterator>(&mut self, bits: R) { for i in bits { self.set(i as usize, true); } } #[must_use] pub fn count_set(&self) -> u8 { let mut count: u8 = 0; for i in 0..T::BIT_LENGTH { if self.get(i) { count += 1; } } count } /// Display bits that are set. /// +1 is added to the display, so it matches the cell values (1..N) pub fn display(&self) -> String { self.iter() .map(|i| (i+1).to_string()) // .map(|i| i.to_string()) .collect::>().join(",") } } impl fmt::Debug for GenBits { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { let result = self.iter() // .map(|i| (i+1).to_string()) .map(|i| i.to_string()) .collect::>().join(","); if result.is_empty() { return write!(f, "{{}}") } else { return write!(f, "<{}>", result) } } } pub struct GenBitsIterator<'a, T: Integer + BitField> { pub possible: &'a GenBits, pub index: usize, } impl GenBits { pub fn iter(&self) -> GenBitsIterator { GenBitsIterator { possible: self, index: 0, } } } impl<'a, T: Integer + BitField> Iterator for GenBitsIterator<'a, T> { type Item = u8; fn next(&mut self) -> Option { while (self.index < T::BIT_LENGTH) && (!self.possible.get(self.index as usize)) { self.index += 1; } if self.index == T::BIT_LENGTH { None } else { self.index += 1; Some((self.index - 1) as u8) } } } #[cfg(test)] mod tests { // use crate::sudoku::*; use crate::bits::*; #[test] fn check_u16() { let mut p = GenBits::(0); p.clear(); for i in 0..9 { let mut result = p.get(i); assert_eq!(result, false); p.set(i, true); result = p.get(i); assert_eq!(result, true); } p = GenBits::(0); p.set(3, true); p.set(5, true); p.set(6, true); assert_eq!(3, p.count_set()); let values: Vec = p.iter().collect(); assert_eq!(values, vec!(3, 5, 6)); assert_eq!(3, p.count_set()); p.set(0, true); assert_eq!(4, p.count_set()); p = GenBits::(0); p.set_bits(0..6); for i in 0..6 { let result = p.get(i); assert_eq!(result, true); } assert_eq!(p.get(6), false); } #[test] fn check_u32() { let mut p = GenBits::(0); p.clear(); for i in 0..29 { let mut result = p.get(i); assert_eq!(result, false); p.set(i, true); result = p.get(i); assert_eq!(result, true); } p = GenBits::(0); p.set(13, true); p.set(15, true); p.set(26, true); assert_eq!(3, p.count_set()); let values: Vec = p.iter().collect(); assert_eq!(values, vec!(13, 15, 26)); assert_eq!(3, p.count_set()); p.set(0, true); assert_eq!(4, p.count_set()); p = GenBits::(0); p.set_bits(10..26); for i in 10..26 { let result = p.get(i); assert_eq!(result, true); } assert_eq!(p.get(9), false); assert_eq!(p.get(27), false); } }