top of page

Advanced Python Functions: Scope, Nesting, Arguments, and Practical Data Processing Applications



Mastering Python Functions and Scopes: A Comprehensive Guide


Hello and welcome to this comprehensive guide on mastering Python functions and scopes. Whether you are a beginner or have some experience in Python, this tutorial aims to deepen your understanding of Python functions and scopes. By the end of this tutorial, you will not only be able to comfortably define and use your own Python functions but also understand and navigate the nuances of Python scopes like a pro. So, without further ado, let's get started!


1. Introduction to User-Defined Functions


A user-defined function in Python is like your personal assistant. Imagine you are a chef, and you have this assistant who is good at chopping vegetables. You just tell them what vegetables to chop and how you want them, and voila, they get the job done for you, saving your time and effort. Now, wouldn't it be cool if you could have such an assistant in your coding journey too? Well, user-defined functions are just that!


In Python, we define a function using the def keyword, followed by the function name and parentheses () which may contain parameters, followed by a colon :. The subsequent lines of code, indented under the function definition, form the function body.


Let's write a simple Python function that acts as our assistant and adds two numbers for us:

def add_numbers(num1, num2):
    result = num1 + num2
    return result

# Now let's use our function to add 7 and 3
print(add_numbers(7, 3))

Output:

10


We passed two parameters num1 and num2 to the function add_numbers. The function calculated the sum and then return keyword sent the result back. We then printed the result which, as expected, is 10.


2. Understanding Scope in Functions


Scope in Python is the part of the program where a variable is accessible. Consider scope as different rooms in a house. If you left your keys in the kitchen, you could not find them in the living room or bedroom because they are in the scope of the kitchen. The concept of scope in Python works similarly. Let's explore the different types of scope in Python.

  • Global scope: Names defined in the main body of a script or a Python program are said to be in the global scope. They are accessible from anywhere in the code.

  • Local scope: Names defined within a function have local scope. They cease to exist after the function execution, just like a chocolate bar in your room ceases to exist (from your room) once you eat it!

  • Built-in scope: Python has some pre-defined names (like pre-installed furniture in a rented apartment) in its built-in module, and they are in the built-in scope.

Here is an example that demonstrates global and local scope:

# Global scope
x = 10

def show_scope():
    # Local scope
    y = 5
    print("Inside function: ", x, y)

show_scope()

# This will give error as y is not accessible outside the function (local scope)
print("Outside function: ", x, y)

The output of the above code:

Inside function:  10 5
---------------------------------------------------------------------------
NameError                                 Traceback (most recent call last)
<ipython-input-2-2b4d44ee35ed> in <module>()
     10
     11 # This will give error as y is not accessible outside the function (local scope)
---> 12 print("Outside function: ", x, y)

NameError: name 'y' is not defined


The error message at the end demonstrates that y does not exist outside of the function scope.


As for built-in scope, functions like print(), len(), and type() are part of Python's built-in scope. You don't have to define these functions before using them because Python has already done it for you.


Diving Deeper into Python Functions and Scopes


Welcome back! We've already covered the basics of user-defined functions and an introduction to scopes in Python. Now, let's explore further and understand the interplay between global and local scopes, how the global keyword comes into play, and dive into the exciting world of nested functions.


3. Global vs. Local Scope


Understanding the interaction between global and local scopes can be compared to understanding the relationship between universal laws and local customs. Just like local customs can't change universal laws, local variables cannot alter global variables in Python unless explicitly permitted.

Python has a hierarchy to look up variables: it first looks in the local scope (inside any function that is currently being executed), then the global scope, and finally the built-in scope. Let's illustrate this with an example:

# Global variable
x = "global"

def check_scope():
    # Local variable with the same name
    x = "local"
    print("Inside function: ", x)

check_scope()
print("Outside function: ", x)

Output:

Inside function:  local
Outside function:  global


Here, x inside the function did not overwrite x in the global scope. This is because Python treats them as two different variables - one in the global scope and one in

the local scope of the function check_scope.


4. The Global Keyword


But what if you want to break this rule and modify a global variable from within a function? Just like you'd need special permission to modify universal laws, Python provides the global keyword.


When we declare a variable as global in a function, it references the global variable, not creating a new local one. Let's modify the previous example:

# Global variable
x = "global"

def modify_global():
    global x
    # This now refers to the global variable x
    x = "local"
    print("Inside function: ", x)

modify_global()
print("Outside function: ", x)

Output:

Inside function:  local
Outside function:  local


As you can see, the global keyword allowed us to modify the global variable x from within the function.


5. Introduction to Nested Functions


A nested function is a function defined inside another function, like a box inside another box. It's like having a smaller kitchen inside your main kitchen specifically for baking, so you can organize and manage your baking tools separately.


Here's an example of a nested function:

def outer_function(x):
    # Nested function
    def inner_function(y):
        return y ** 2
    # The outer function returns the square of x
    return inner_function(x)

print(outer_function(5))

Output:

25


The outer_function takes an input x, passes it to the inner_function, which squares it, and then returns the result.


Understanding Nested Functions and Scope Rules in Python


Let's continue our journey through the world of Python functions and scopes. In this part, we will discuss how a function can return another function, dive into the use of the nonlocal keyword and finally understand the LEGB rule in Python.


6. Returning Functions


Continuing from the previous part, consider our example of a nested function. The outer_function returns the result of calling the inner_function. But what if it returned the inner_function itself?


Imagine you are ordering a custom cake. Instead of getting a ready-made cake, you get a personal baker who can make you any cake you want. Similarly, returning a function gives us flexibility.


Here's an example:

def outer_function(x):
    # Nested function
    def inner_function(y):
        return y ** x
    # The outer function returns the inner function itself
    return inner_function

# Creating a function that squares the input
square = outer_function(2)
print(square(5))

Output:

25


Here, outer_function returned a function inner_function that raises its input to the power of x. We created a squaring function by setting x to 2. This concept, where the returned function "remembers" the values from the enclosing scope, is known as 'closure' in computer science.


7. The Nonlocal Keyword


Remember how we used the global keyword to modify global variables from a function? Similarly, we can use the nonlocal keyword to modify variables from an enclosing scope in a nested function.


Consider our nested function example. What if the inner_function wants to modify x?

def outer_function():
    x = 2
    # Nested function
    def inner_function(y):
        nonlocal x
        x = y ** x
        return x
    # Execute the inner function
    return inner_function(5)

print(outer_function())

Output:

32


With the nonlocal keyword, inner_function was able to modify x from the enclosing outer_function.


8. LEGB Rule


Python's mechanism to look up variable names is known as the LEGB rule, which stands for Local, Enclosing, Global, and Built-in scopes.

  1. Local: Names defined in the current function.

  2. Enclosing: Names in the local scope of any and all enclosing functions, from inner to outer.

  3. Global: Names assigned at the top-level of a module, or declared global in a def within the file.

  4. Built-in: Pre-assigned in the built-in names module.


It's like Python first looking in your personal bag (Local), then your friend's bag (Enclosing), then your teacher's bag (Global), and finally the school's lost-and-found (Built-in) to find your lost pen.


Default, Flexible Arguments and Practical Data Processing in Python


Now that we have a solid understanding of scopes and nested functions, let's shift our focus towards the world of data manipulation with Python functions. In this part, we will explore default and flexible arguments, and finally, apply our knowledge to a practical data processing task.


9. Default and Flexible Arguments


While designing a custom car, there are certain components you can choose - like color, seats, and wheels. However, some options are pre-selected for you by the manufacturer - like the engine, the base design, etc. In Python, we can design functions similarly - with some default arguments (pre-selected by you, the function creator), and some that the user can choose.


For example:

def power(base, exponent=2):
    return base ** exponent

# With one argument
print(power(2))  # Squares the input
# With two arguments
print(power(2, 3))  # Cubes the input

Output:

4
8


Flexible arguments allow us to pass any number of arguments to a function, just like a buffet dinner where you can choose any number of dishes you want. We use *args for positional arguments and **kwargs for keyword arguments:

def buffet(*args, **kwargs):
    for arg in args:
        print(f"I want {arg}")
    for kw in kwargs:
        print(f"I want {kw} to be {kwargs[kw]}")

buffet("pasta", "pizza", drink="cola", dessert="ice cream")

Output:

I want pasta
I want pizza
I want drink to be cola
I want dessert to be ice cream


10. Practical Application: Data Processing


Now let's apply this to a data processing task. Imagine you have a DataFrame and you want to count occurrences in a specific column. But let's make it more generic - we want to pass any number of column names to our function.


Here's how we might do it:

import pandas as pd

# Assume df is your DataFrame
df = pd.DataFrame({
    'col1': ['apple', 'banana', 'apple', 'apple', 'banana', 'apple'],
    'col2': ['orange', 'orange', 'banana', 'banana', 'orange', 'orange'],
    'col3': ['banana', 'apple', 'orange', 'apple', 'apple', 'banana']
})

def count_values(df, *args):
    result = {}
    for arg in args:
        result[arg] = df[arg].value_counts().to_dict()
    return result

print(count_values(df, 'col1', 'col2'))

Output:

{
  'col1': {'apple': 4, 'banana': 2},
  'col2': {'orange': 4, 'banana': 2}
}


This function counts the occurrences in the given columns and returns a dictionary. It's flexible - you can pass any number of column names, thanks to flexible arguments. And you can use it with any DataFrame, making it a versatile tool in your data processing arsenal.


Conclusion


We have traversed the landscape of Python functions and scopes, delving into nested functions, global and nonlocal keywords, default and flexible arguments, and practical data processing tasks. We've seen how these elements can be put together to create powerful and flexible tools in Python. Understanding these concepts is essential to effective programming and data manipulation in Python.


It's like being a carpenter - knowing what each tool does and when to use it makes you effective at building things. Keep practicing and you'll be a Python carpentry expert in no time!

bottom of page