TheAlgorithms-C-Plus-Plus/search/fibonacci_search.cpp

114 lines
3.1 KiB
C++
Raw Normal View History

2020-07-07 21:27:45 +08:00
/**
* Copyright 2020 @author sprintyaf
* @file fibonacci_search.cpp
2020-07-07 21:27:45 +08:00
* @brief [Fibonacci search
* algorithm](https://en.wikipedia.org/wiki/Fibonacci_search_technique)
*/
2020-07-07 18:38:58 +08:00
#include <iostream>
2020-07-07 21:27:45 +08:00
#include <vector> // for std::vector class
#include <cassert> // for assert
#include <cstdlib> // for random numbers
#include <algorithm> // for sorting
2020-07-07 21:27:45 +08:00
2020-07-07 18:38:58 +08:00
2020-07-07 21:27:45 +08:00
/**
* @brief using fibonacci search algorithm finds an index of a given element in a sorted array
*
* @param arr sorted array
* @param value value that we're looking for
* @returns if the array contains the value, returns an index of the element. otherwise -1.
2020-07-07 21:27:45 +08:00
*/
int fibonacci_search(const std::vector<int> &arr, int value){
2020-07-07 18:38:58 +08:00
int last = 0, current = 1, offset = -1, index;
int length = arr.size();
2020-07-07 18:38:58 +08:00
int next = last + current;
while(next < length){
last = current;
current = next;
next = last + current;
}
while(next > 1){
index = std::min(offset + last, length-1);
2020-07-07 18:38:58 +08:00
if(arr[index] < value){
next = current;
current = last;
last = next - current;
offset = index;
} else if(arr[index] > value){
next = last;
current = current - last;
last = next - current;
} else {
return index;
}
}
if(current && !arr.empty() && arr[offset+1] == value){
2020-07-07 18:38:58 +08:00
return offset+1;
}
return -1;
}
2020-07-07 21:27:45 +08:00
/**
* @brief random tests for checking performance when an array doesn't contain an element
*/
bool no_occurence_tests(){
2020-07-07 21:27:45 +08:00
bool passed = true;
int rand_num, rand_value, index, num_tests = 1000;
std::vector<int> arr;
while(num_tests--){
arr.clear();
for(int i = 0; i < 100; i++){
2020-07-08 00:45:03 +08:00
rand_num = std::rand() % 1000;
arr.push_back(rand_num);
}
2020-07-08 00:45:03 +08:00
rand_value = std::rand() % 1000;
while(std::find(arr.begin(), arr.end(), rand_value) != arr.end()){
std::remove(arr.begin(), arr.end(), rand_value);
}
sort(arr.begin(), arr.end());
index = fibonacci_search(arr, rand_value);
passed = passed && (index == -1);
}
2020-07-07 21:27:45 +08:00
return passed;
}
/**
* @brief random tests which cover cases when we have one, multiple or zero occurences of the value we're looking for
*/
bool random_tests(){
2020-07-07 21:27:45 +08:00
bool passed = true;
int rand_num, rand_value, index, real_value, num_tests = 10000;
std::vector<int> arr;
while(num_tests--){
arr.clear();
for(int i = 0; i < 100; i++){
2020-07-08 00:45:03 +08:00
rand_num = std::rand() % 1000;
arr.push_back(rand_num);
}
2020-07-08 00:45:03 +08:00
rand_value = std::rand() % 1000;
std::sort(arr.begin(), arr.end());
index = fibonacci_search(arr, rand_value);
if(index != -1){
real_value = arr[index];
passed = passed && (real_value == rand_value);
} else {
passed = passed && (std::find(arr.begin(), arr.end(), rand_value) == arr.end());
}
}
2020-07-07 21:27:45 +08:00
return passed;
}
/**
* Main Function
* testing the algorithm
*/
int main() {
assert(no_occurence_tests());
assert(random_tests());
2020-07-07 18:38:58 +08:00
return 0;
}