| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | |
| | #[derive(Clone, Debug, PartialEq)] |
| | pub struct Point { |
| | dims: Vec<f32>, |
| | } |
| |
|
| | impl Point { |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | pub fn new(dims: Vec<f32>) -> Self { |
| | Self { dims } |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | pub fn origin(dims: usize) -> Self { |
| | Self { |
| | dims: vec![0.0; dims], |
| | } |
| | } |
| |
|
| | |
| | pub fn dimensionality(&self) -> usize { |
| | self.dims.len() |
| | } |
| |
|
| | |
| | pub fn dims(&self) -> &[f32] { |
| | &self.dims |
| | } |
| |
|
| | |
| | pub fn dims_mut(&mut self) -> &mut [f32] { |
| | &mut self.dims |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | pub fn magnitude(&self) -> f32 { |
| | self.dims.iter().map(|x| x * x).sum::<f32>().sqrt() |
| | } |
| |
|
| | |
| | pub fn is_normalized(&self) -> bool { |
| | let mag = self.magnitude(); |
| | (mag - 1.0).abs() < 0.001 |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | pub fn normalize(&self) -> Self { |
| | let mag = self.magnitude(); |
| | if mag == 0.0 { |
| | return self.clone(); |
| | } |
| | Self { |
| | dims: self.dims.iter().map(|x| x / mag).collect(), |
| | } |
| | } |
| |
|
| | |
| | pub fn add(&self, other: &Point) -> Self { |
| | assert_eq!( |
| | self.dimensionality(), |
| | other.dimensionality(), |
| | "Points must have same dimensionality" |
| | ); |
| | Self { |
| | dims: self |
| | .dims |
| | .iter() |
| | .zip(other.dims.iter()) |
| | .map(|(a, b)| a + b) |
| | .collect(), |
| | } |
| | } |
| |
|
| | |
| | pub fn scale(&self, scalar: f32) -> Self { |
| | Self { |
| | dims: self.dims.iter().map(|x| x * scalar).collect(), |
| | } |
| | } |
| | } |
| |
|
| | #[cfg(test)] |
| | mod tests { |
| | use super::*; |
| |
|
| | #[test] |
| | fn test_new_point() { |
| | let p = Point::new(vec![1.0, 2.0, 3.0]); |
| | assert_eq!(p.dimensionality(), 3); |
| | assert_eq!(p.dims(), &[1.0, 2.0, 3.0]); |
| | } |
| |
|
| | #[test] |
| | fn test_origin() { |
| | let origin = Point::origin(768); |
| | assert_eq!(origin.dimensionality(), 768); |
| | assert!(origin.dims().iter().all(|&x| x == 0.0)); |
| | } |
| |
|
| | #[test] |
| | fn test_magnitude() { |
| | let p = Point::new(vec![3.0, 4.0]); |
| | assert!((p.magnitude() - 5.0).abs() < 0.0001); |
| | } |
| |
|
| | #[test] |
| | fn test_normalize() { |
| | let p = Point::new(vec![3.0, 4.0]); |
| | let normalized = p.normalize(); |
| | assert!(normalized.is_normalized()); |
| | assert!((normalized.dims()[0] - 0.6).abs() < 0.0001); |
| | assert!((normalized.dims()[1] - 0.8).abs() < 0.0001); |
| | } |
| |
|
| | #[test] |
| | fn test_normalize_zero() { |
| | let p = Point::origin(3); |
| | let normalized = p.normalize(); |
| | assert_eq!(normalized.dims(), &[0.0, 0.0, 0.0]); |
| | } |
| |
|
| | #[test] |
| | fn test_add() { |
| | let a = Point::new(vec![1.0, 2.0]); |
| | let b = Point::new(vec![3.0, 4.0]); |
| | let c = a.add(&b); |
| | assert_eq!(c.dims(), &[4.0, 6.0]); |
| | } |
| |
|
| | #[test] |
| | fn test_scale() { |
| | let p = Point::new(vec![1.0, 2.0]); |
| | let scaled = p.scale(2.0); |
| | assert_eq!(scaled.dims(), &[2.0, 4.0]); |
| | } |
| |
|
| | #[test] |
| | #[should_panic(expected = "same dimensionality")] |
| | fn test_add_different_dims_panics() { |
| | let a = Point::new(vec![1.0, 2.0]); |
| | let b = Point::new(vec![1.0, 2.0, 3.0]); |
| | let _ = a.add(&b); |
| | } |
| | } |
| |
|