Binary Search implemented in C++












1















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.










share|improve this question





























    1















    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.










    share|improve this question



























      1












      1








      1








      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.










      share|improve this question
















      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






      share|improve this question















      share|improve this question













      share|improve this question




      share|improve this question








      edited Jan 30 '17 at 23:43







      user2635139

















      asked Jan 30 '17 at 23:22









      user2635139user2635139

      1471211




      1471211






















          2 Answers
          2






          active

          oldest

          votes


















          4














          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.






          share|improve this answer

































            0














            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).






            share|improve this answer























              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
              });


              }
              });














              draft saved

              draft discarded


















              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









              4














              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.






              share|improve this answer






























                4














                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.






                share|improve this answer




























                  4












                  4








                  4







                  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.






                  share|improve this answer















                  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.







                  share|improve this answer














                  share|improve this answer



                  share|improve this answer








                  edited 2 mins ago

























                  answered Jan 31 '17 at 0:26









                  Jerry CoffinJerry Coffin

                  27.9k460126




                  27.9k460126

























                      0














                      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).






                      share|improve this answer




























                        0














                        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).






                        share|improve this answer


























                          0












                          0








                          0







                          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).






                          share|improve this answer













                          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).







                          share|improve this answer












                          share|improve this answer



                          share|improve this answer










                          answered Jan 31 '17 at 18:38









                          Babra CunninghamBabra Cunningham

                          19311




                          19311






























                              draft saved

                              draft discarded




















































                              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.




                              draft saved


                              draft discarded














                              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





















































                              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







                              Popular posts from this blog

                              Morgemoulin

                              Scott Moir

                              Souastre