Binary Search implemented in C++
This is my implementation of a recursive binary search in C++. I'm still a newbie at C++ so:
- Have I used any bad practices?
- How could the code be improved upon in terms of efficiency?
- Could recursion cause a stack overflow error, and if so how can I avoid this?
Criticism is welcome and appreciated!
#include<iostream>
#include<sstream>
#include<vector>
template <typename T>
void outputVector(std::vector<T> input){
std::cout << "[";
for(int i = 0; i < input.size(); i++){
std::cout << input[i];
if(i != input.size()-1){
std::cout << ", ";
}
}
std::cout << "]n";
}
int binarySearch(std::vector<double> input, double target, int startIndex = 0){
if(input.size() < 2){
//std::cout << "Closest match foundn";
return startIndex;
}
int middleIndex = int(input.size())/2;
double middle = input[middleIndex];
if(target == middle){
return middleIndex + startIndex;
}
else if(target < middle){
std::vector<double>::const_iterator begin = input.begin();
std::vector<double>::const_iterator last = input.begin() + middleIndex;
std::vector<double> firstHalf(begin, last);
//std::cout << "First Half: ";
//outputVector(firstHalf);
return binarySearch(firstHalf, target, startIndex);
}
else if(target > middle){
std::vector<double>::const_iterator begin = input.begin() + middleIndex + 1;
std::vector<double>::const_iterator last = input.begin() + input.size();
std::vector<double> secondHalf(begin, last);
//std::cout << "Second Half: ";
//outputVector(secondHalf);
return binarySearch(secondHalf, target, startIndex+middleIndex+1);
}
else{
return -1;
}
}
std::vector<double> doubleVectorInput(){
std::string inputString;
getline(std::cin, inputString);
std::vector<double> array;
std::istringstream iss(inputString);
float val;
while(iss >> val){
array.push_back(val);
}
return array;
}
int main(){
double target;
std::cout << "Vector to search: ";
std::vector<double> input = doubleVectorInput();
std::cout << "Target: "; std::cin >> target;
std::cout << "Target index: " << binarySearch(input, target) << "n";
return 0;
}
Thank you.
c++ beginner c++11 binary-search
add a comment |
This is my implementation of a recursive binary search in C++. I'm still a newbie at C++ so:
- Have I used any bad practices?
- How could the code be improved upon in terms of efficiency?
- Could recursion cause a stack overflow error, and if so how can I avoid this?
Criticism is welcome and appreciated!
#include<iostream>
#include<sstream>
#include<vector>
template <typename T>
void outputVector(std::vector<T> input){
std::cout << "[";
for(int i = 0; i < input.size(); i++){
std::cout << input[i];
if(i != input.size()-1){
std::cout << ", ";
}
}
std::cout << "]n";
}
int binarySearch(std::vector<double> input, double target, int startIndex = 0){
if(input.size() < 2){
//std::cout << "Closest match foundn";
return startIndex;
}
int middleIndex = int(input.size())/2;
double middle = input[middleIndex];
if(target == middle){
return middleIndex + startIndex;
}
else if(target < middle){
std::vector<double>::const_iterator begin = input.begin();
std::vector<double>::const_iterator last = input.begin() + middleIndex;
std::vector<double> firstHalf(begin, last);
//std::cout << "First Half: ";
//outputVector(firstHalf);
return binarySearch(firstHalf, target, startIndex);
}
else if(target > middle){
std::vector<double>::const_iterator begin = input.begin() + middleIndex + 1;
std::vector<double>::const_iterator last = input.begin() + input.size();
std::vector<double> secondHalf(begin, last);
//std::cout << "Second Half: ";
//outputVector(secondHalf);
return binarySearch(secondHalf, target, startIndex+middleIndex+1);
}
else{
return -1;
}
}
std::vector<double> doubleVectorInput(){
std::string inputString;
getline(std::cin, inputString);
std::vector<double> array;
std::istringstream iss(inputString);
float val;
while(iss >> val){
array.push_back(val);
}
return array;
}
int main(){
double target;
std::cout << "Vector to search: ";
std::vector<double> input = doubleVectorInput();
std::cout << "Target: "; std::cin >> target;
std::cout << "Target index: " << binarySearch(input, target) << "n";
return 0;
}
Thank you.
c++ beginner c++11 binary-search
add a comment |
This is my implementation of a recursive binary search in C++. I'm still a newbie at C++ so:
- Have I used any bad practices?
- How could the code be improved upon in terms of efficiency?
- Could recursion cause a stack overflow error, and if so how can I avoid this?
Criticism is welcome and appreciated!
#include<iostream>
#include<sstream>
#include<vector>
template <typename T>
void outputVector(std::vector<T> input){
std::cout << "[";
for(int i = 0; i < input.size(); i++){
std::cout << input[i];
if(i != input.size()-1){
std::cout << ", ";
}
}
std::cout << "]n";
}
int binarySearch(std::vector<double> input, double target, int startIndex = 0){
if(input.size() < 2){
//std::cout << "Closest match foundn";
return startIndex;
}
int middleIndex = int(input.size())/2;
double middle = input[middleIndex];
if(target == middle){
return middleIndex + startIndex;
}
else if(target < middle){
std::vector<double>::const_iterator begin = input.begin();
std::vector<double>::const_iterator last = input.begin() + middleIndex;
std::vector<double> firstHalf(begin, last);
//std::cout << "First Half: ";
//outputVector(firstHalf);
return binarySearch(firstHalf, target, startIndex);
}
else if(target > middle){
std::vector<double>::const_iterator begin = input.begin() + middleIndex + 1;
std::vector<double>::const_iterator last = input.begin() + input.size();
std::vector<double> secondHalf(begin, last);
//std::cout << "Second Half: ";
//outputVector(secondHalf);
return binarySearch(secondHalf, target, startIndex+middleIndex+1);
}
else{
return -1;
}
}
std::vector<double> doubleVectorInput(){
std::string inputString;
getline(std::cin, inputString);
std::vector<double> array;
std::istringstream iss(inputString);
float val;
while(iss >> val){
array.push_back(val);
}
return array;
}
int main(){
double target;
std::cout << "Vector to search: ";
std::vector<double> input = doubleVectorInput();
std::cout << "Target: "; std::cin >> target;
std::cout << "Target index: " << binarySearch(input, target) << "n";
return 0;
}
Thank you.
c++ beginner c++11 binary-search
This is my implementation of a recursive binary search in C++. I'm still a newbie at C++ so:
- Have I used any bad practices?
- How could the code be improved upon in terms of efficiency?
- Could recursion cause a stack overflow error, and if so how can I avoid this?
Criticism is welcome and appreciated!
#include<iostream>
#include<sstream>
#include<vector>
template <typename T>
void outputVector(std::vector<T> input){
std::cout << "[";
for(int i = 0; i < input.size(); i++){
std::cout << input[i];
if(i != input.size()-1){
std::cout << ", ";
}
}
std::cout << "]n";
}
int binarySearch(std::vector<double> input, double target, int startIndex = 0){
if(input.size() < 2){
//std::cout << "Closest match foundn";
return startIndex;
}
int middleIndex = int(input.size())/2;
double middle = input[middleIndex];
if(target == middle){
return middleIndex + startIndex;
}
else if(target < middle){
std::vector<double>::const_iterator begin = input.begin();
std::vector<double>::const_iterator last = input.begin() + middleIndex;
std::vector<double> firstHalf(begin, last);
//std::cout << "First Half: ";
//outputVector(firstHalf);
return binarySearch(firstHalf, target, startIndex);
}
else if(target > middle){
std::vector<double>::const_iterator begin = input.begin() + middleIndex + 1;
std::vector<double>::const_iterator last = input.begin() + input.size();
std::vector<double> secondHalf(begin, last);
//std::cout << "Second Half: ";
//outputVector(secondHalf);
return binarySearch(secondHalf, target, startIndex+middleIndex+1);
}
else{
return -1;
}
}
std::vector<double> doubleVectorInput(){
std::string inputString;
getline(std::cin, inputString);
std::vector<double> array;
std::istringstream iss(inputString);
float val;
while(iss >> val){
array.push_back(val);
}
return array;
}
int main(){
double target;
std::cout << "Vector to search: ";
std::vector<double> input = doubleVectorInput();
std::cout << "Target: "; std::cin >> target;
std::cout << "Target index: " << binarySearch(input, target) << "n";
return 0;
}
Thank you.
c++ beginner c++11 binary-search
c++ beginner c++11 binary-search
edited Jan 30 '17 at 23:43
user2635139
asked Jan 30 '17 at 23:22
user2635139user2635139
1471211
1471211
add a comment |
add a comment |
2 Answers
2
active
oldest
votes
Space Complexity
This is quite inefficient in terms of the storage space it uses.
Assuming the bisection works as we'd hope, and the vector is cut exactly in half every time, the recursive call copies half the input vector. If, for example, we started with an array of a million items, the first recursive call will copy half a million items. The second recursive call will copy a quarter of a million items.
Since the first call also passed the input array by value, we expect that our search of one million items ends up copying about two million items (in addition to the original vector the user passed).
Time Complexity
Copying those items takes a fair amount of time as well. To be more specific, it takes linear time.
Summary
We normally expect a binary search to have complexity of about $O(log N)$, and only descend to $O(N)$ in relatively rare worst cases. In this case, we get $O(N)$ in about the best case, and $O(N^2)$ in the bad cases.
This violates the normal expectations of a binary search badly enough that it effectively only barely qualifies as a binary search at all. In many cases, we can expect its performance to be substantially worse than a simple linear search.
Recommendation
You really don't want to pass the vector by value. Either pass it by reference, along with a pair of indexes into the vector giving the current upper and lower bounds of the part you want to search, or else pass a pair of iterators (there are probably other options as well, but in any case you want to pass an indication of the data to search rather than copying all the data you're going to search).
There are other parts of the code that could use changing as well, but that set of changes is likely to render many (most?) of them obsolete anyway, so I'm not going to get into other areas for now.
add a comment |
For an algorithm to be classified as a binary search it must have worse-case time complexity of O(log n).
As mentioned by Jerry, here you have a worse case O(N ^2), I suggest you re-think your method.
I'm not sure why you'd perform this operation recursively in C++ (not a great language for such an approach).
add a comment |
Your Answer
StackExchange.ifUsing("editor", function () {
return StackExchange.using("mathjaxEditing", function () {
StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
});
});
}, "mathjax-editing");
StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "196"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f154026%2fbinary-search-implemented-in-c%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
2 Answers
2
active
oldest
votes
2 Answers
2
active
oldest
votes
active
oldest
votes
active
oldest
votes
Space Complexity
This is quite inefficient in terms of the storage space it uses.
Assuming the bisection works as we'd hope, and the vector is cut exactly in half every time, the recursive call copies half the input vector. If, for example, we started with an array of a million items, the first recursive call will copy half a million items. The second recursive call will copy a quarter of a million items.
Since the first call also passed the input array by value, we expect that our search of one million items ends up copying about two million items (in addition to the original vector the user passed).
Time Complexity
Copying those items takes a fair amount of time as well. To be more specific, it takes linear time.
Summary
We normally expect a binary search to have complexity of about $O(log N)$, and only descend to $O(N)$ in relatively rare worst cases. In this case, we get $O(N)$ in about the best case, and $O(N^2)$ in the bad cases.
This violates the normal expectations of a binary search badly enough that it effectively only barely qualifies as a binary search at all. In many cases, we can expect its performance to be substantially worse than a simple linear search.
Recommendation
You really don't want to pass the vector by value. Either pass it by reference, along with a pair of indexes into the vector giving the current upper and lower bounds of the part you want to search, or else pass a pair of iterators (there are probably other options as well, but in any case you want to pass an indication of the data to search rather than copying all the data you're going to search).
There are other parts of the code that could use changing as well, but that set of changes is likely to render many (most?) of them obsolete anyway, so I'm not going to get into other areas for now.
add a comment |
Space Complexity
This is quite inefficient in terms of the storage space it uses.
Assuming the bisection works as we'd hope, and the vector is cut exactly in half every time, the recursive call copies half the input vector. If, for example, we started with an array of a million items, the first recursive call will copy half a million items. The second recursive call will copy a quarter of a million items.
Since the first call also passed the input array by value, we expect that our search of one million items ends up copying about two million items (in addition to the original vector the user passed).
Time Complexity
Copying those items takes a fair amount of time as well. To be more specific, it takes linear time.
Summary
We normally expect a binary search to have complexity of about $O(log N)$, and only descend to $O(N)$ in relatively rare worst cases. In this case, we get $O(N)$ in about the best case, and $O(N^2)$ in the bad cases.
This violates the normal expectations of a binary search badly enough that it effectively only barely qualifies as a binary search at all. In many cases, we can expect its performance to be substantially worse than a simple linear search.
Recommendation
You really don't want to pass the vector by value. Either pass it by reference, along with a pair of indexes into the vector giving the current upper and lower bounds of the part you want to search, or else pass a pair of iterators (there are probably other options as well, but in any case you want to pass an indication of the data to search rather than copying all the data you're going to search).
There are other parts of the code that could use changing as well, but that set of changes is likely to render many (most?) of them obsolete anyway, so I'm not going to get into other areas for now.
add a comment |
Space Complexity
This is quite inefficient in terms of the storage space it uses.
Assuming the bisection works as we'd hope, and the vector is cut exactly in half every time, the recursive call copies half the input vector. If, for example, we started with an array of a million items, the first recursive call will copy half a million items. The second recursive call will copy a quarter of a million items.
Since the first call also passed the input array by value, we expect that our search of one million items ends up copying about two million items (in addition to the original vector the user passed).
Time Complexity
Copying those items takes a fair amount of time as well. To be more specific, it takes linear time.
Summary
We normally expect a binary search to have complexity of about $O(log N)$, and only descend to $O(N)$ in relatively rare worst cases. In this case, we get $O(N)$ in about the best case, and $O(N^2)$ in the bad cases.
This violates the normal expectations of a binary search badly enough that it effectively only barely qualifies as a binary search at all. In many cases, we can expect its performance to be substantially worse than a simple linear search.
Recommendation
You really don't want to pass the vector by value. Either pass it by reference, along with a pair of indexes into the vector giving the current upper and lower bounds of the part you want to search, or else pass a pair of iterators (there are probably other options as well, but in any case you want to pass an indication of the data to search rather than copying all the data you're going to search).
There are other parts of the code that could use changing as well, but that set of changes is likely to render many (most?) of them obsolete anyway, so I'm not going to get into other areas for now.
Space Complexity
This is quite inefficient in terms of the storage space it uses.
Assuming the bisection works as we'd hope, and the vector is cut exactly in half every time, the recursive call copies half the input vector. If, for example, we started with an array of a million items, the first recursive call will copy half a million items. The second recursive call will copy a quarter of a million items.
Since the first call also passed the input array by value, we expect that our search of one million items ends up copying about two million items (in addition to the original vector the user passed).
Time Complexity
Copying those items takes a fair amount of time as well. To be more specific, it takes linear time.
Summary
We normally expect a binary search to have complexity of about $O(log N)$, and only descend to $O(N)$ in relatively rare worst cases. In this case, we get $O(N)$ in about the best case, and $O(N^2)$ in the bad cases.
This violates the normal expectations of a binary search badly enough that it effectively only barely qualifies as a binary search at all. In many cases, we can expect its performance to be substantially worse than a simple linear search.
Recommendation
You really don't want to pass the vector by value. Either pass it by reference, along with a pair of indexes into the vector giving the current upper and lower bounds of the part you want to search, or else pass a pair of iterators (there are probably other options as well, but in any case you want to pass an indication of the data to search rather than copying all the data you're going to search).
There are other parts of the code that could use changing as well, but that set of changes is likely to render many (most?) of them obsolete anyway, so I'm not going to get into other areas for now.
edited 2 mins ago
answered Jan 31 '17 at 0:26
Jerry CoffinJerry Coffin
27.9k460126
27.9k460126
add a comment |
add a comment |
For an algorithm to be classified as a binary search it must have worse-case time complexity of O(log n).
As mentioned by Jerry, here you have a worse case O(N ^2), I suggest you re-think your method.
I'm not sure why you'd perform this operation recursively in C++ (not a great language for such an approach).
add a comment |
For an algorithm to be classified as a binary search it must have worse-case time complexity of O(log n).
As mentioned by Jerry, here you have a worse case O(N ^2), I suggest you re-think your method.
I'm not sure why you'd perform this operation recursively in C++ (not a great language for such an approach).
add a comment |
For an algorithm to be classified as a binary search it must have worse-case time complexity of O(log n).
As mentioned by Jerry, here you have a worse case O(N ^2), I suggest you re-think your method.
I'm not sure why you'd perform this operation recursively in C++ (not a great language for such an approach).
For an algorithm to be classified as a binary search it must have worse-case time complexity of O(log n).
As mentioned by Jerry, here you have a worse case O(N ^2), I suggest you re-think your method.
I'm not sure why you'd perform this operation recursively in C++ (not a great language for such an approach).
answered Jan 31 '17 at 18:38
Babra CunninghamBabra Cunningham
19311
19311
add a comment |
add a comment |
Thanks for contributing an answer to Code Review Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
Use MathJax to format equations. MathJax reference.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f154026%2fbinary-search-implemented-in-c%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown