What are popular interview questions in Python

author image Hirely
at 05 Jan, 2025

Question: What are Python generators?

Answer:

Python generators are a type of iterable, like lists or tuples, but instead of storing all the values in memory, they generate values on the fly and yield them one by one when needed. This allows you to work with large datasets or infinite sequences without consuming a lot of memory.

Generators are defined using functions with the yield keyword or by using generator expressions.


Key Features of Generators:

  1. Lazy Evaluation: Generators compute values only when needed, which means they can represent infinite sequences or very large datasets.
  2. Memory Efficiency: Since they yield one item at a time, generators do not require the entire dataset to be stored in memory.
  3. State Preservation: A generator function maintains its state between calls, so it can resume where it left off.

Creating Generators:

There are two main ways to create generators in Python:


1. Using a Generator Function (with yield):

A generator function is defined like a normal function but uses the yield keyword to return values. Each time the generator’s __next__() method is called (or next() is used), the function executes until it hits a yield statement, returns the value, and then suspends execution.

Example:

def count_up_to(max):
    count = 1
    while count <= max:
        yield count  # Yield the current value and pause execution
        count += 1

# Create a generator
counter = count_up_to(5)

# Use next() to get values
print(next(counter))  # Output: 1
print(next(counter))  # Output: 2
print(next(counter))  # Output: 3
  • Each time next(counter) is called, the function resumes from where it left off.
  • When there are no more values to yield, the generator raises a StopIteration exception.

2. Using a Generator Expression:

You can also create generators using a syntax similar to list comprehensions but with parentheses instead of square brackets.

Example:

squares = (x * x for x in range(1, 6))

# The generator does not calculate values until requested
for square in squares:
    print(square)

Output:

1
4
9
16
25
  • This is more memory-efficient than a list comprehension because it doesn’t generate all the values upfront.

How Generators Work:

  • yield: When the generator’s function is called, it doesn’t execute all at once. Instead, it pauses at the yield expression, returns the yielded value, and remembers where it left off. When you call next(), execution resumes from the last yield point.
  • StopIteration: When there are no more values to generate, a generator raises the StopIteration exception, signaling that the iteration is complete.

Benefits of Using Generators:

  1. Memory Efficiency: Generators only yield values when needed, meaning they don’t store the entire sequence in memory.
  2. Lazy Evaluation: Ideal for working with large datasets or infinite sequences (e.g., reading lines from a large file).
  3. State Preservation: Generators preserve their state between iterations, making it easy to track the progress of iterations.

Example: Generating Infinite Sequences:

Generators are particularly useful for generating infinite sequences, as they don’t require storing the entire sequence in memory.

def infinite_sequence():
    num = 0
    while True:
        yield num
        num += 1

gen = infinite_sequence()

# Get the first 5 numbers of the infinite sequence
for _ in range(5):
    print(next(gen))

Output:

0
1
2
3
4

In this example, the generator will continue to produce numbers indefinitely, but it never stores the entire sequence in memory at once.


Using yield vs return:

  • yield: Pauses the function and remembers its state, allowing it to resume later.
  • return: Ends the function completely, returning the value and not maintaining any state.

Summary of Generator Characteristics:

  • Lazy evaluation: Values are produced only when requested.
  • Memory efficient: Values are not stored in memory.
  • Stateful: The function’s state is preserved between calls to next().

Generators are especially useful when working with large datasets, infinite sequences, or when you want to implement custom iteration logic in an efficient way.

Question: What is the difference between shallow copy and deep copy in Python?

Answer:

In Python, copying refers to creating a duplicate of an object, but the way the copy behaves can differ significantly. The two main types of copying are shallow copy and deep copy.


1. Shallow Copy:

A shallow copy creates a new object, but it does not create copies of objects that are contained within the original object. Instead, it copies references to the inner objects. This means that if the original object contains other objects (e.g., lists, dictionaries, or custom objects), the shallow copy will refer to the same inner objects as the original.

Characteristics:

  • Creates a new object, but does not create copies of objects within the original object.
  • Changes to mutable objects inside the copy affect the original object, because they both refer to the same inner objects.
  • The outer object is a new object, but inner objects are shared between the original and the copied object.

Example:

import copy

original = [[1, 2, 3], [4, 5, 6]]
shallow_copy = copy.copy(original)

# Modify an inner list in the shallow copy
shallow_copy[0][0] = 999

print("Original:", original)        # Output: [[999, 2, 3], [4, 5, 6]]
print("Shallow Copy:", shallow_copy) # Output: [[999, 2, 3], [4, 5, 6]]

In the above example:

  • The outer list is copied, but the inner lists [1, 2, 3] and [4, 5, 6] are not copied. Both original and shallow_copy refer to the same inner lists.
  • Modifying an inner list in the shallow copy also affects the original because they share the same references to the inner lists.

2. Deep Copy:

A deep copy creates a new object and recursively copies all objects found in the original object. This means that not only is the outer object copied, but all the objects within it are also copied recursively, so the copied object does not share any references with the original object.

Characteristics:

  • Creates a completely independent copy of the original object, including all objects contained within it.
  • Changes to the copied object do not affect the original object.
  • More memory and time-intensive because it recursively copies all elements.

Example:

import copy

original = [[1, 2, 3], [4, 5, 6]]
deep_copy = copy.deepcopy(original)

# Modify an inner list in the deep copy
deep_copy[0][0] = 999

print("Original:", original)        # Output: [[1, 2, 3], [4, 5, 6]]
print("Deep Copy:", deep_copy)      # Output: [[999, 2, 3], [4, 5, 6]]

In this case:

  • The deep copy creates a new copy of both the outer list and all inner lists. Therefore, changes to the inner lists in the deep copy do not affect the original object.

Key Differences:

FeatureShallow CopyDeep Copy
Copying ProcessCopies the outer object, but inner objects are referenced.Copies the outer object and all inner objects.
Object ReferencesInner objects are shared between the original and copy.Inner objects are copied, so no references are shared.
Effect of ModificationModifying mutable objects inside the copy affects the original.Modifying objects inside the copy does not affect the original.
PerformanceFaster and uses less memory because it doesn’t copy inner objects.Slower and uses more memory due to copying nested objects recursively.
Use CaseSuitable when the original and copy can share some internal objects.Suitable when you need a completely independent copy.

When to Use Each:

  • Shallow copy: Use when you want a copy of the object but can share the internal objects between the original and copied objects. It’s faster and more memory-efficient.
  • Deep copy: Use when you need a completely independent copy of the original object, and you do not want any shared references between the original and the copy, especially when the object contains nested mutable objects.

How to Perform Copying in Python:

  • Shallow copy: Use copy.copy() or the slicing operator for lists (new_list = old_list[:]).
  • Deep copy: Use copy.deepcopy().

Example for Shallow Copy:

import copy
shallow_copy = copy.copy(original)

Example for Deep Copy:

import copy
deep_copy = copy.deepcopy(original)

Read More

If you can’t get enough from this article, Aihirely has plenty more related information, such as Python interview questions, Python interview experiences, and details about various Python job positions. Click here to check it out.

Related Posts

Trace Job opportunities

Hirely, your exclusive interview companion, empowers your competence and facilitates your interviews.

Get Started Now