13. Returning functions from functions

Quiz: 0/5

Returning functions from functions is a powerful and flexible feature in Python that opens up a world of possibilities for creating higher-order functions. This technique is widely used in advanced programming concepts such as closures, decorators, and factory functions. Let's explore this concept in detail, examining its workings, benefits, and practical examples.

What does it mean to return functions from functions?

In Python, functions are first-class objects, meaning they can be passed around, assigned to variables, and even returned from other functions. A function that returns another function is known as a higher-order function. This feature allows for powerful programming patterns like closures and dynamic function generation.

The ability to return functions is useful for:

  • Encapsulating behavior: By defining a function inside another function, you can create reusable and customizable behavior.
  • Generating customized functions: You can create functions on the fly, dynamically adjusting their behavior based on parameters.
  • Creating closures: Functions returned from other functions can retain access to variables from the outer function's scope, enabling the management of state.

Why return functions from functions?

Returning functions provides several advantages:

  • Encapsulation: It allows you to bundle a specific behavior within a function, which can then be customized and reused elsewhere.
  • Code reusability: Functions that generate other functions can help eliminate repetitive code, leading to cleaner and more modular code.
  • Maintaining state: When combined with closures, returning functions enables you to maintain state across multiple calls, which is ideal for scenarios like counters or caching.

Example: Basic function returning another function

Here’s a simple example where a function returns another function:

def outer_function(message):
    """Outer function that returns an inner function."""
    def inner_function():
        return f"Message: {message}"
    return inner_function

# Create a new function using the outer function
new_function = outer_function("Hello, World!")

# Call the returned function
print(new_function())  # Output: Message: Hello, World!

Explanation:

  • The outer_function takes a message as an argument and defines an inner function inner_function that returns a string containing the message.
  • When outer_function is called, it returns the inner_function, which can then be invoked later.

This example demonstrates how you can return a function dynamically, allowing you to execute it whenever needed.

Example: Creating custom functions with parameters

Returning functions can also be used to create customizable functions with specific parameters:

def power_of(n):
    """Returns a function that raises a number to the power of n."""
    def exponent(base):
        return base ** n
    return exponent

# Create functions to square and cube numbers
square = power_of(2)
cube = power_of(3)

# Using the created functions
print(square(5))  # Output: 25
print(cube(5))    # Output: 125

Explanation:

  • The power_of function returns a function exponent that takes a base number and raises it to the power of n.
  • By calling power_of(2), we generate a function that squares numbers, and by calling power_of(3), we generate a function that cubes numbers.

This example highlights how functions can be dynamically created based on the parameters passed to the outer function.

Example: Closures

A closure is a function that retains access to the variables from its enclosing scope even after the outer function has finished executing. Here’s an example of a closure:

def make_counter():
    """Returns a function that counts from a starting value."""
    count = 0

    def counter():
        nonlocal count  # Access the count variable from the outer scope
        count += 1
        return count

    return counter

# Create a counter function
my_counter = make_counter()

# Call the counter multiple times
print(my_counter())  # Output: 1
print(my_counter())  # Output: 2
print(my_counter())  # Output: 3

Explanation:

  • The make_counter function initializes a variable count and defines an inner function counter that increments count each time it is called.
  • When we call my_counter(), the closure remembers the value of count and continues from where it left off.

This example demonstrates how closures can be used to maintain state across function calls, which is particularly useful for things like counters or managing resources.

Benefits of returning functions

Returning functions offers several advantages:

  • Modular and reusable code: You can create small, reusable function components that can be dynamically customized based on input parameters.
  • Increased flexibility: Functions can be generated on the fly, making your code more adaptable to changing requirements.
  • State management: Closures allow for the management of state across multiple invocations, enabling more sophisticated behaviors in your programs.

Summary

Returning functions from functions is a powerful and flexible feature in Python. It enables you to:

  • Encapsulate behaviors and create customizable functions.
  • Dynamically generate functions based on input parameters, making your code more flexible and reusable.
  • Maintain state across function calls using closures, which is essential in many advanced programming scenarios.

 

Hands-On Practise

Exercise: Creating a custom multiplier

Objective:
Create a function that returns a new function which multiplies its argument by a specified number. The returned function should be customizable, so that it can multiply by different values based on the input to the outer function.

Task:

  1. Write a function create_multiplier(factor) that takes a number factor and returns a function.
  2. The returned function should take a single argument and multiply it by factor.
  3. Create two multiplier functions: one that multiplies by 5, and another that multiplies by 10.
  4. Use these multiplier functions to multiply different numbers.
Output:

Quizzes: 0/5

Question 1:

What does it mean to return a function from another function in Python?

  • You can define a function inside another function and return it.
  • It’s only possible to return functions from built-in functions.
  • Functions can only return values, not other functions.
  • Returning functions is only possible when using lambda functions.

Question 2:

What is the purpose of closures when returning functions from functions?

  • Closures are used to maintain state across multiple calls of the returned function.
  • Closures make functions return random values each time they are called.
  • Closures prevent functions from accepting parameters.
  • Closures are only used to modify global variables.

Question 3:

Which of the following is a benefit of returning functions from other functions in Python?

  • It allows for more complex code that’s harder to debug.
  • It allows for encapsulation and dynamic function generation.
  • It can only be used with built-in functions like map() and filter() .
  • It forces functions to become anonymous.

Question 4:

In the provided example of returning functions, what does the make_counter function return?

  • A function that increments a counter each time it’s called.
  • A function that resets the counter each time it’s called.
  • A string that counts the number of calls.
  • A counter variable that can be accessed directly.

Question 5:

Which of the following is an example of creating a customized function using returned functions?

  • Returning a function that multiplies a number by a given factor.
  • Returning a function that only accepts numbers greater than 10.
  • Returning a function that prints the current time.
  • Returning a function that generates random numbers.

Feedback

Share your thoughts about this lesson

Loading...
Previous lessonPassing functions as arguments