Square Root Calculation Methods in C++

by | C++, DSA, Programming, Searching Algorithms

This guide will go through how to calculate square roots in C++ using the built-in sqrt() function, Newton’s method, and binary search. This guide provides practical examples, performance benchmarks, and insights to choose the best method for your needs.

Introduction

Calculating square roots is a fundamental operation in many programming applications, from geometric calculations to numerical analysis. While C++ provides built-in functions for this purpose, understanding different implementation methods can be valuable for both learning and specialized applications. In this guide, we’ll explore various approaches to calculating square roots in C++.

💡 Key Insight: While the built-in sqrt() function is optimized for most use cases, understanding alternative implementations can provide valuable insights into numerical methods and algorithm design.

Built-in Square Root Function

The simplest way to calculate square roots in C++ is using the standard library’s sqrt() function:

Basic Square Root Calculation Using std::sqrt
#include <iostream>
#include <cmath>

int main() {
    double number = 16.0;
    double result = sqrt(number);

    std::cout << "Square root of " << number << " is " << result << std::endl;

    // Handle negative numbers
    double negative = -16.0;
    if (negative < 0) {
        std::cout << "Square root of " << negative << " is undefined for real numbers" << std::endl;
    }

    return 0;
}
Square root of 16 is 4
Square root of -16 is undefined for real numbers

The built-in function is highly optimized and suitable for most applications. However, let's explore how we can implement our own square root calculations for educational purposes or specialized needs.

Newton's Method Implementation

Newton's method (also known as the Newton-Raphson method) is an efficient way to calculate square roots iteratively. The method uses the following formula:

\[ x_{n+1} = \frac{1}{2}\left(x_n + \frac{S}{x_n}\right) \]

where:

  • \( S \): The number whose square root we want to find
  • \( x_n \): Current approximation
  • \( x_{n+1} \): Next approximation
Newton's Method Implementation for Square Root
#include <iostream>
#include <cmath>

double newtonSqrt(double number, double epsilon = 1e-10) {
    if (number < 0) {
        return std::nan(""); // Return NaN for negative numbers
    }
    if (number == 0) {
        return 0;
    }

    double x = number; // Initial guess
    double previous;

    do {
        previous = x;
        x = 0.5 * (x + number / x); // Newton's method formula
    } while (std::abs(x - previous) > epsilon);

    return x;
}

int main() {
    double numbers[] = {16.0, 2.0, 100.0, 0.25};

    for (double num : numbers) {
        double result = newtonSqrt(num);
        std::cout << "Square root of " << num << " is " << result
                  << " (Error: " << std::abs(result - std::sqrt(num)) << ")"
                  << std::endl;
    }

    return 0;
}
Square root of 16 is 4 (Error: 0)
Square root of 2 is 1.41421 (Error: 2.22045e-16)
Square root of 100 is 10 (Error: 0)
Square root of 0.25 is 0.5 (Error: 0)

Binary Search Method

Another approach is using binary search to find the square root. While not as efficient as Newton's method, it's intuitive and demonstrates important algorithmic concepts:

Binary Search Method for Square Root Calculation
#include <iostream>
#include <cmath>

double binarySearchSqrt(double number, double epsilon = 1e-10) {
    if (number < 0) {
        return std::nan("");
    }
    if (number == 0) {
        return 0;
    }

    double low = 0;
    double high = number > 1 ? number : 1;
    double mid;

    while (high - low > epsilon) {
        mid = (low + high) / 2;
        double square = mid * mid;

        if (square > number) {
            high = mid;
        } else {
            low = mid;
        }
    }

    return mid;
}

int main() {
    double numbers[] = {16.0, 2.0, 100.0, 0.25};

    for (double num : numbers) {
        double result = binarySearchSqrt(num);
        std::cout << "Square root of " << num << " is " << result
                  << " (Error: " << std::abs(result - std::sqrt(num)) << ")"
                  << std::endl;
    }

    return 0;
}
Square root of 16 is 4 (Error: 5.82077e-11)
Square root of 2 is 1.41421 (Error: 4.7055e-11)
Square root of 100 is 10 (Error: 5.45697e-11)
Square root of 0.25 is 0.5 (Error: 5.82077e-11)

Performance Comparison

Performance is a critical factor when choosing the best method to compute square roots, especially in applications requiring high efficiency or frequent calculations. Each method—whether it's the built-in sqrt() function, Newton's iterative method, or binary search—has its own strengths and trade-offs. The built-in function is typically the fastest due to hardware-level optimizations, while custom implementations offer greater control and flexibility for specialized use cases.

In this section, we will evaluate the performance of these methods using a practical benchmarking approach. By measuring execution times over a large number of iterations, we can compare their speed and efficiency under identical conditions. This analysis will help identify the most suitable approach for various scenarios, from general-purpose applications to situations where precision or algorithmic understanding takes priority.

Complete Performance Comparison Implementation
#include <iostream>
#include <chrono>
#include <cmath>

// Newton's method implementation
double newtonSqrt(double number, double epsilon = 1e-10) {
    if (number < 0) {
        return std::nan("");
    }
    if (number == 0) {
        return 0;
    }

    double x = number;
    double previous;

    do {
        previous = x;
        x = 0.5 * (x + number / x);
    } while (std::abs(x - previous) > epsilon);

    return x;
}

// Binary search implementation
double binarySearchSqrt(double number, double epsilon = 1e-10) {
    if (number < 0) {
        return std::nan("");
    }
    if (number == 0) {
        return 0;
    }

    double low = 0;
    double high = number > 1 ? number : 1;
    double mid;

    while (high - low > epsilon) {
        mid = (low + high) / 2;
        double square = mid * mid;

        if (square > number) {
            high = mid;
        } else {
            low = mid;
        }
    }

    return mid;
}

int main() {
    const int iterations = 1000000;
    double number = 16777216.0;  // 2^24

    // Test built-in sqrt
    auto start = std::chrono::high_resolution_clock::now();
    for (int i = 0; i < iterations; i++) {
        volatile double result = std::sqrt(number);
    }
    auto end = std::chrono::high_resolution_clock::now();
    auto built_in_time = std::chrono::duration_cast<std::chrono::microseconds>
                        (end - start).count();

    // Test Newton's method
    start = std::chrono::high_resolution_clock::now();
    for (int i = 0; i < iterations; i++) {
        volatile double result = newtonSqrt(number);
    }
    end = std::chrono::high_resolution_clock::now();
    auto newton_time = std::chrono::duration_cast<std::chrono::microseconds>
                      (end - start).count();

    // Test binary search method
    start = std::chrono::high_resolution_clock::now();
    for (int i = 0; i < iterations; i++) {
        volatile double result = binarySearchSqrt(number);
    }
    end = std::chrono::high_resolution_clock::now();
    auto binary_time = std::chrono::duration_cast<std::chrono::microseconds>
                      (end - start).count();

    std::cout << "Performance comparison for " << iterations << " iterations:\n";
    std::cout << "Built-in sqrt(): " << built_in_time << " microseconds\n";
    std::cout << "Newton's method: " << newton_time << " microseconds\n";
    std::cout << "Binary search: " << binary_time << " microseconds\n";

    return 0;
}
Performance comparison for 1000000 iterations:
Built-in sqrt(): 1739 microseconds
Newton's method: 131035 microseconds
Binary search: 420637 microseconds
Performance tested on:
- CPU Model: Intel(R) Core(TM) i7-9750H @ 2.60GHz
- Cores: 6 physical, 12 logical
- Base Clock Speed: 2.6 GHz

💡 Performance Insight: The built-in sqrt() function is typically the fastest option as it's often implemented using hardware-specific optimizations. Newton's method comes second, while binary search is generally the slowest but most straightforward to implement.

Understanding the Performance Gap

Looking at our performance results, we can see significant differences between the three methods:

  • Built-in sqrt() (1,739 μs): Benefits from hardware-level implementation and CPU-specific optimizations, often using dedicated floating-point instructions.
  • Newton's method (131,035 μs): Converges quadratically, typically requiring only 4-5 iterations for double precision.
  • Binary search (420,637 μs): Shows the poorest performance due to several factors:

The binary search method is particularly slow because:

  1. Fixed Step Size: Unlike Newton's method, which adapts its step size based on the function's behavior, binary search always halves the interval. This means it needs more iterations to achieve the same precision.
  2. Precision Requirements: To achieve the same precision as Newton's method, binary search needs approximately log₂(1/ε) iterations, where ε is our desired precision. For double precision, this means around 50-60 iterations, compared to Newton's 4-5 iterations.
  3. More Operations per Iteration: Each iteration requires multiplication for the midpoint calculation and comparison operations, adding computational overhead.

💡 Teaching Value: Despite its slower performance, binary search remains valuable as a teaching tool because it demonstrates important concepts in numerical methods and showcases how a simple, intuitive approach can solve complex problems.

Practical Considerations

When implementing square root calculations in C++, consider the following factors:

  • Precision Requirements: The built-in sqrt() function provides double precision, which is sufficient for most applications. Custom implementations can be tuned for different precision needs.
  • Performance Needs: For performance-critical applications, the built-in function is usually the best choice due to hardware optimization.
  • Edge Cases: Always handle negative numbers and zero appropriately. Consider using nan() or throwing exceptions for invalid inputs.
  • Numerical Stability: Be aware of potential overflow or underflow issues when dealing with very large or small numbers.

💡 Best Practice: Unless you have specific requirements that necessitate a custom implementation, prefer the standard library's sqrt() function. It's well-tested, optimized, and handles edge cases correctly.

Conclusion

Understanding different methods for calculating square roots not only helps in specific implementation needs but also provides valuable insights into numerical methods and algorithm design. While the built-in sqrt() function is the go-to choice for most applications, knowing how to implement custom solutions can be valuable for educational purposes or specialized requirements.

Congratulations on reading to the end of this tutorial. See the Further Reading section for more documentation and related article.

Further Reading

  • Methods of Computing Square Roots

    A comprehensive overview of various square root computation methods throughout history.

  • C++ Reference: std::sqrt

    Official C++ reference documentation for the square root function.

  • Newton's Method

    Detailed explanation of Newton's method and its applications beyond square root calculation.

  • Binary Search

    An in-depth look at the binary search algorithm, its implementation, and applications.

Attribution and Citation

If you found this guide and tools helpful, feel free to link back to this page or cite it in your work!

Profile Picture
Senior Advisor, Data Science | [email protected] |  + posts

Suf is a senior advisor in data science with deep expertise in Natural Language Processing, Complex Networks, and Anomaly Detection. Formerly a postdoctoral research fellow, he applied advanced physics techniques to tackle real-world, data-heavy industry challenges. Before that, he was a particle physicist at the ATLAS Experiment of the Large Hadron Collider. Now, he’s focused on bringing more fun and curiosity to the world of science and research online.

Buy Me a Coffee ✨