How to Write the Fibonacci Sequence in Python

by | DSA, Python, Tips

Introduction

The Fibonacci sequence is a fascinating mathematical series where each number is the sum of the two preceding ones. Let’s explore different ways to implement this sequence in Python, comparing their efficiency and use cases.

The Fibonacci sequence starts with 0 and 1. Each subsequent number is the sum of the previous two numbers, forming a sequence like this:

0, 1, 1, 2, 3, 5, 8, 13, 21, 34, …

The Fibonacci sequence can be defined mathematically as:

\[ F(n) = F(n-1) + F(n-2), \quad \text{where:} \]

\[ F(0) = 0 \quad \text{and} \quad F(1) = 1 \]

1. Recursive Implementation

One of the simplest and most intuitive ways to calculate the Fibonacci sequence is through recursion. In this approach, the function calls itself with smaller values of n until it reaches the base cases (0 and 1). Here’s the code:

Recursive Fibonacci Implementation
def fibonacci_recursive(n):
    if n <= 1:
        return n
    return fibonacci_recursive(n-1) + fibonacci_recursive(n-2)

# Usage
print([fibonacci_recursive(i) for i in range(10)])
# Output: [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Note: While elegant, this implementation has a significant drawback. Its time complexity is O(2^n), as each function call generates two more calls, leading to an exponential growth in the number of calls. This makes it inefficient for calculating large Fibonacci numbers.

For instance, calculating F(50) would involve trillions of function calls, which is impractical. More optimized approaches are needed for better performance.

2. Dynamic Programming Approach

The recursive approach is elegant but inefficient for larger values of n due to its exponential time complexity. We can significantly improve performance using dynamic programming by storing intermediate results (memoization). This reduces redundant calculations and improves time complexity to O(n).

Dynamic Programming Implementation
def fibonacci_dp(n):
    if n <= 1:
        return n

    dp = [0] * (n + 1)
    dp[1] = 1

    for i in range(2, n + 1):
        dp[i] = dp[i-1] + dp[i-2]

    return dp[n]

# Usage
print([fibonacci_dp(i) for i in range(10)])
# Output: [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Key Points:

  • This approach uses a list, dp, to store Fibonacci numbers up to n.
  • Each Fibonacci number is computed once and stored, ensuring no redundant calculations.
  • The time complexity is O(n), as we iterate through the sequence only once.
  • The space complexity is O(n), as we use a list to store all Fibonacci numbers up to n.

Note: While this implementation is efficient, the space complexity can be further reduced by only storing the last two numbers in the sequence, which we'll cover in the next section.

3. Space-Optimized Iterative Solution

The dynamic programming approach is efficient in terms of time complexity but can still be optimized for space. Instead of storing the entire sequence, we can calculate Fibonacci numbers using only two variables. This reduces the space complexity from O(n) to O(1).

Iterative Implementation
def fibonacci_iterative(n):
    if n <= 1:
        return n

    a, b = 0, 1
    for _ in range(2, n + 1):
        a, b = b, a + b
    return b

# Usage
print([fibonacci_iterative(i) for i in range(10)])
# Output: [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Key Points:

  • We maintain two variables, a and b, to track the last two numbers in the sequence.
  • In each iteration, we update these variables to compute the next Fibonacci number.
  • The time complexity remains O(n), as we calculate each Fibonacci number exactly once.
  • The space complexity is reduced to O(1), as no additional memory is used for storing the sequence.

Note: This implementation is optimal in terms of both time and space. It is ideal for calculating Fibonacci numbers in scenarios where memory is constrained.

4. Generator Implementation

When working with large Fibonacci sequences, storing all values in memory can become impractical. Instead, we can use a generator to produce Fibonacci numbers on demand. This approach leverages Python's yield keyword to create an efficient, memory-friendly implementation.

Generator Implementation
def fibonacci_generator():
    a, b = 0, 1
    while True:
        yield a
        a, b = b, a + b

# Usage
fib = fibonacci_generator()
first_10 = [next(fib) for _ in range(10)]
print(first_10)
# Output: [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Key Points:

  • Generators are ideal for iterating over potentially infinite sequences, as they compute values lazily (on demand).
  • This implementation uses the yield keyword to return Fibonacci numbers one at a time.
  • The time complexity is O(1) per number generated, and the space complexity is effectively O(1), as no additional storage is required beyond the two variables, a and b.
  • You can stop the generator at any point, making it highly flexible for various use cases.

Note: Generators are particularly useful when you need to process or calculate Fibonacci numbers in a streaming fashion or when memory usage needs to be minimized.

For instance, you can use this generator to calculate Fibonacci numbers up to a specific condition or process them in real-time in an application.

5. Class-Based Implementation

A class-based implementation provides a structured way to encapsulate the logic of generating Fibonacci numbers. By implementing the iterator protocol, we can create an iterable Fibonacci sequence with a defined maximum length.

Class-Based Implementation
class FibonacciSequence:
    def __init__(self, max_length):
        self.max_length = max_length

    def __iter__(self):
        self.n1 = 0
        self.n2 = 1
        self.count = 0
        return self

    def __next__(self):
        if self.count >= self.max_length:
            raise StopIteration

        if self.count == 0:
            self.count += 1
            return self.n1
        elif self.count == 1:
            self.count += 1
            return self.n2

        result = self.n1 + self.n2
        self.n1 = self.n2
        self.n2 = result
        self.count += 1
        return result

# Usage
fib_sequence = FibonacciSequence(10)
print(list(fib_sequence))
# Output: [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]

Key Points:

  • The FibonacciSequence class implements the __iter__ and __next__ methods to conform to Python's iterator protocol.
  • The __iter__ method initializes the variables for tracking the Fibonacci sequence.
  • The __next__ method calculates the next number in the sequence, raising a StopIteration exception when the maximum length is reached.
  • This approach allows the sequence to be used in for loops or converted directly to a list, making it flexible for various use cases.

Note: A class-based implementation is particularly useful when you want to add additional functionality or state management, such as tracking statistics about the sequence or modifying behavior dynamically.

It also makes the code more modular and reusable in larger projects.

Performance Comparison

Understanding the performance differences between various Fibonacci implementations is crucial for selecting the most efficient approach for a given use case. Below is a simple benchmark script to compare the recursive, dynamic programming, and iterative implementations.

Benchmark Code
import time

def benchmark(func, n):
    start = time.time()
    result = func(n)
    end = time.time()
    return end - start

n = 35
print(f"Recursive: {benchmark(fibonacci_recursive, n):.9f} seconds")
print(f"Dynamic Programming: {benchmark(fibonacci_dp, n):.9f} seconds")
print(f"Iterative: {benchmark(fibonacci_iterative, n):.9f} seconds")

Benchmark Results

Below are the benchmark results for calculating F(35) using different implementations of the Fibonacci sequence:

Execution Times:

  • Recursive: 1.369483948 seconds
  • Dynamic Programming: 0.000005960 seconds
  • Iterative: 0.000003099 seconds

Key Insights:

  • The Recursive implementation is exponentially slower due to redundant calculations.
  • The Dynamic Programming approach is highly efficient, reducing the execution time significantly by storing intermediate results.
  • The Iterative approach is the fastest and most memory-efficient, outperforming all other methods.

Note: When benchmarking, the runtime for each function may vary slightly depending on the system and environment. The performance gap widens as n increases, especially for the recursive implementation.

For n > 40, the recursive approach may take minutes or more, while the iterative and dynamic programming solutions remain efficient.

Practical Applications

The Fibonacci sequence has numerous real-world applications, from approximating mathematical constants like the Golden Ratio to analyzing properties of the sequence itself. Below are two practical examples demonstrating its utility.

Golden Ratio Approximation
def calculate_golden_ratio(n):
    fib = fibonacci_iterative(n)
    prev_fib = fibonacci_iterative(n-1)
    return fib / prev_fib

# As n increases, this ratio approaches the golden ratio (≈ 1.618034)
print(calculate_golden_ratio(100))
1.618033988749895

Golden Ratio:

  • The ratio of consecutive Fibonacci numbers, F(n) / F(n-1), approaches the Golden Ratio (≈ 1.618034) as n increases.
  • This relationship is a cornerstone in understanding proportions in nature, architecture, and art.
  • For example, many plants grow leaves, seeds, or petals in Fibonacci-like patterns to optimize space and sunlight exposure.
Sequence Analysis
def analyze_sequence(n):
    sequence = [fibonacci_iterative(i) for i in range(n)]
    even_sum = sum(x for x in sequence if x % 2 == 0)
    odd_sum = sum(x for x in sequence if x % 2 != 0)
    return {
        'sequence': sequence,
        'even_sum': even_sum,
        'odd_sum': odd_sum,
        'ratio': even_sum / odd_sum if odd_sum != 0 else 0
    }

print(analyze_sequence(10))
{'sequence': [0, 1, 1, 2, 3, 5, 8, 13, 21, 34],
'even_sum': 44,
'odd_sum': 44,
'ratio': 1.0}

Sequence Analysis:

  • This analysis computes the sum of even and odd Fibonacci numbers up to n terms and their ratio.
  • It demonstrates how Fibonacci numbers can be analyzed for their statistical and mathematical properties.
  • Applications include understanding patterns in sequences and applying them to solve problems in cryptography, data science, and algorithm design.

Note: The Fibonacci sequence is a gateway to exploring deeper mathematical concepts, including number theory and optimization. Its presence in natural phenomena, art, and science makes it a versatile tool for discovery and learning.

Conclusion

Congratulations on reading to the end of this tutorial! You have explored various implementations of the Fibonacci sequence in Python, including iterative, recursive, dynamic programming, generator-based, and class-based approaches. These implementations illustrate key Python programming concepts:

  • Recursive vs. Iterative Approaches: Understanding the trade-offs between clarity and performance.
  • Dynamic Programming: Optimizing performance by storing intermediate results.
  • Generators and Iterators: Using lazy evaluation for memory-efficient solutions.
  • Object-Oriented Programming: Creating modular and reusable code with class-based designs.
  • Performance Optimization: Benchmarking and choosing the best approach for your specific requirements.

The Fibonacci sequence and Fibonacci numbers are ubiquitous in nature and underpin fascinating mathematical expressions. Learning how to implement the Fibonacci sequence is particularly helpful for understanding how recursion works in Python.

Choose the Implementation That Best Fits Your Needs:

  • Iterative Approach: For best performance and minimal memory usage.
  • Generator: Ideal for efficiently generating large sequences without overloading memory.
  • Class Implementation: Offers a Pythonic and modular solution for advanced use cases.
  • Recursive Approach: Best suited for educational purposes or when simplicity and clarity are the main goals.

Important Note: When dealing with extremely large Fibonacci numbers, consider using Python's decimal module for higher precision, or take steps to handle potential integer overflow issues in other languages.

To learn how to implement the Fibonacci sequence in C++, visit the article: How to Write the Fibonacci Sequence in C++.

For further reading on recursive functions, check out the article: How to Find the Factorial of a Number in Python.

The Fibonacci sequence is more than just a mathematical curiosity. Its properties and patterns make it a versatile tool for exploring fundamental programming and mathematical concepts. Whether you’re learning to code, analyzing data, or solving optimization problems, the Fibonacci sequence provides endless opportunities for exploration.

If you found this guide helpful, feel free to link back to this post for attribution and share it with others exploring logistic regression in PyTorch!

HTML: Attribution: The Research Scientist Pod - How to Write the Fibonacci Sequence in Python

Markdown: [The Research Scientist Pod - ](https://researchdatapod.com/how-to-write-the-fibonacci-sequence-in-python/)

Have fun and happy researching!

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 ✨