-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinterpolation_search.rs
57 lines (51 loc) · 1.63 KB
/
interpolation_search.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
pub fn interpolation_search<Ordering>(nums: &[i32], item: &i32) -> Result<usize, usize> {
// early check
if nums.is_empty() {
return Err(0);
}
let mut low: usize = 0;
let mut high: usize = nums.len() - 1;
while low <= high {
if *item < nums[low] || *item > nums[high] {
break;
}
let offset: usize = low
+ (((high - low) / (nums[high] - nums[low]) as usize) * (*item - nums[low]) as usize);
match nums[offset].cmp(&*item) {
std::cmp::Ordering::Equal => return Ok(offset),
std::cmp::Ordering::Less => low = offset + 1,
std::cmp::Ordering::Greater => high = offset - 1,
}
}
Err(0)
}
#[cfg(test)]
mod tests {
use super::*;
use std::cmp::Ordering;
#[test]
fn returns_err_if_empty_slice() {
let nums = [];
assert_eq!(interpolation_search::<Ordering>(&nums, &3), Err(0));
}
#[test]
fn returns_err_if_target_not_found() {
let nums = [1, 2, 3, 4, 5, 6];
assert_eq!(interpolation_search::<Ordering>(&nums, &10), Err(0));
}
#[test]
fn returns_first_index() {
let index: Result<usize, usize> = interpolation_search::<Ordering>(&[1, 2, 3, 4, 5], &1);
assert_eq!(index, Ok(0));
}
#[test]
fn returns_last_index() {
let index: Result<usize, usize> = interpolation_search::<Ordering>(&[1, 2, 3, 4, 5], &5);
assert_eq!(index, Ok(4));
}
#[test]
fn returns_middle_index() {
let index: Result<usize, usize> = interpolation_search::<Ordering>(&[1, 2, 3, 4, 5], &3);
assert_eq!(index, Ok(2));
}
}