Python 3.x Interview Questions

author image Hirely
at 10 Jan, 2025

Question: What is the difference between @staticmethod and @classmethod in Python?

Answer:

In Python, both @staticmethod and @classmethod are used to define methods that are not bound to a specific instance of a class, but they differ in how they interact with the class and its instances.


1. @staticmethod (Static Method)

A static method is a method that belongs to the class itself, rather than an instance of the class. It does not take self or cls as the first parameter. This means that static methods cannot access or modify the instance (object) or the class itself. They are essentially regular functions that happen to reside inside a class for organizational purposes.

  • Does not have access to the instance (self) or the class (cls).
  • Does not modify class state or instance state.
  • It is used when you need a method that logically belongs to the class but doesn’t need access to class or instance data.

Example of @staticmethod:

class Math:
    @staticmethod
    def add(a, b):
        return a + b

# Call the static method without creating an instance
result = Math.add(5, 10)
print(result)  # Output: 15

In this example:

  • The add() method does not access or modify any class or instance properties. It simply performs an operation and returns a result.

2. @classmethod (Class Method)

A class method is a method that is bound to the class rather than an instance of the class. It takes the class itself (cls) as the first parameter, not the instance (self). Class methods can access and modify the class state, but not the instance state. Class methods are typically used for factory methods or methods that affect the class as a whole (rather than individual instances).

  • Takes cls (the class) as the first parameter (not self).
  • Can access and modify class state but not instance state.
  • Can be called on the class itself or on an instance, but they always refer to the class.

Example of @classmethod:

class Dog:
    species = "Canis familiaris"  # Class variable

    def __init__(self, name, age):
        self.name = name
        self.age = age

    @classmethod
    def get_species(cls):
        return cls.species

# Call the class method on the class itself
print(Dog.get_species())  # Output: Canis familiaris

# Call the class method on an instance
dog = Dog("Buddy", 5)
print(dog.get_species())  # Output: Canis familiaris

In this example:

  • The get_species() method accesses the class-level variable species using cls.
  • You can call a class method on both the class itself and on an instance, but it always refers to the class.

Summary of Differences:

Feature@staticmethod@classmethod
First parameterDoes not take self or cls.Takes cls (the class itself) as the first parameter.
Access to instanceCannot access instance variables (self).Cannot access instance variables (self) directly, but can access class variables.
Access to classCannot access class variables (cls).Can access and modify class variables using cls.
Use caseWhen a method doesn’t need to modify the class or instance.When you need to operate on class-level data or create factory methods.
How to callCalled on the class or an instance, but typically used on the class.Called on the class, but can also be called on instances (it always refers to the class).

When to Use:

  • Use @staticmethod: When the method does not need to interact with the class or instance and is just logically part of the class.

  • Use @classmethod: When the method needs to modify the class state, or when you want to create a method that can work with class-level data or return an instance of the class (e.g., a factory method).


Factory Method Example with @classmethod:

A common use of class methods is to implement factory methods, which are methods that return an instance of the class.

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @classmethod
    def from_birth_year(cls, name, birth_year):
        age = 2024 - birth_year  # Assume current year is 2024
        return cls(name, age)

# Creating a Person object using the class method
person = Person.from_birth_year("Alice", 1990)
print(person.name)  # Output: Alice
print(person.age)   # Output: 34

Here, the from_birth_year() class method is used to create a new Person object by calculating the age based on the birth year. This is a typical use case for @classmethod to construct objects in different ways.

Question: How does Python handle memory management?

Answer:

Python manages memory automatically using a combination of techniques, including reference counting, garbage collection, and memory pools. Below is an explanation of each of these methods:


1. Reference Counting

Python uses reference counting as one of the main techniques for memory management. Every object in Python has an associated reference count, which tracks the number of references (variables, data structures, etc.) pointing to that object.

  • When an object is created, its reference count is initialized to 1.
  • When a new reference to the object is created, the reference count is incremented.
  • When a reference is deleted or goes out of scope, the reference count is decremented.

Once an object’s reference count reaches zero, meaning there are no references to that object anymore, Python automatically deallocates the memory used by that object. This is called automatic memory deallocation.

Example:

a = [1, 2, 3]  # a references the list [1, 2, 3]
b = a           # b references the same list
del a           # reference count of the list is 1 (still referenced by b)
del b           # reference count drops to 0, and the object is deallocated

In the above example:

  • The list [1, 2, 3] is initially referenced by a.
  • When b = a is executed, the reference count of the list increases to 2.
  • Deleting a reduces the reference count, but the object is still referenced by b.
  • When b is deleted, the reference count becomes zero, and the object is deallocated.

2. Garbage Collection

Python also includes a garbage collector to manage cyclic references. While reference counting handles most memory deallocation cases, it cannot deal with circular references (e.g., two objects referencing each other). Python’s garbage collector solves this problem.

The garbage collector is a part of the gc module, which is responsible for detecting and cleaning up reference cycles (i.e., groups of objects that reference each other but are not reachable from any external object).

How Garbage Collection Works:

  • The garbage collector periodically checks for unreachable objects (i.e., objects that are no longer referenced by any part of the program, including circular references).
  • If such objects are found, the garbage collector deallocates them and frees their memory.
  • Python’s garbage collector uses a technique called generational garbage collection, which divides objects into three generations based on how long they have been in memory.
Generational Garbage Collection:
  • Young Generation (Generation 0): Newly created objects.
  • Middle Generation (Generation 1): Objects that survived one or more garbage collection cycles.
  • Old Generation (Generation 2): Objects that have survived several cycles and are unlikely to be garbage collected.

This approach optimizes garbage collection by focusing on collecting younger objects, which are more likely to be garbage, and avoiding frequent collection of older, long-lived objects.

Example of Cyclic Garbage Collection:

import gc

class A:
    def __init__(self):
        self.ref = None

# Create two objects that reference each other (cyclic reference)
obj1 = A()
obj2 = A()

obj1.ref = obj2
obj2.ref = obj1

# Delete the objects
del obj1
del obj2

# At this point, the objects are no longer in use, but the reference cycle may still exist.
# We can manually run garbage collection
gc.collect()

In this case:

  • obj1 and obj2 reference each other, creating a cyclic reference.
  • Without garbage collection, these objects would remain in memory because of the circular references.
  • By manually invoking gc.collect(), Python detects and cleans up these unreachable objects.

3. Memory Pools

Python uses memory pools to optimize memory allocation and reduce overhead. Python’s memory management system divides memory into small, fixed-size blocks (or pools), which are used to store objects of similar sizes. This helps avoid fragmentation and speeds up memory allocation and deallocation.

  • Small Object Allocator: For small objects, Python uses a system called pymalloc, which allocates memory in blocks of predefined sizes (e.g., for small integers or short strings). This system reduces memory overhead and improves performance for allocating small objects.
  • Large Objects: For larger objects (e.g., large lists, dictionaries), Python falls back on using the operating system’s allocator.

4. The del Statement

The del statement in Python removes a reference to an object. This reduces the reference count by 1. When the reference count of an object drops to zero, the object is deallocated. However, del does not immediately free memory; it only removes a reference. The memory is reclaimed later, when Python’s garbage collector runs.

Example of del:

a = [1, 2, 3]
del a  # 'a' is deleted, but memory is freed later

In this case, the list [1, 2, 3] is deleted, but its memory is freed by the garbage collector once it detects that there are no references to it.


5. Memory Leaks in Python

While Python has a robust memory management system, memory leaks can still occur, typically due to unintentional references. For example, objects might remain in memory because they are still referenced, even though they are no longer needed. Common causes include:

  • Holding references in global variables or static data structures.
  • Circular references that are not properly cleaned up by the garbage collector.

To help detect memory leaks, you can use Python’s gc module to inspect garbage collection activity and monitor memory usage.


6. Tools for Memory Management

Python provides tools to help manage and monitor memory:

  • gc module: Provides functions for interacting with the garbage collector, manually triggering collection, and debugging memory issues.
  • sys.getsizeof(): A built-in function to determine the size of an object in memory.
  • objgraph: A third-party library for generating object reference graphs to help detect memory leaks.

Example of checking memory usage with sys.getsizeof():

import sys

a = [1, 2, 3]
print(sys.getsizeof(a))  # Outputs the memory size of the object `a` in bytes

Summary of Python’s Memory Management:

  1. Reference Counting: Objects are deallocated when their reference count drops to zero.
  2. Garbage Collection: Python’s garbage collector handles circular references and unreferenced objects using generational garbage collection.
  3. Memory Pools: Small objects are allocated in memory pools to reduce fragmentation and improve performance.
  4. del Statement: Used to delete references, which decreases the reference count.
  5. Memory Leaks: Unused objects can still remain in memory due to unintentional references or circular references that the garbage collector may not handle.

Python’s memory management system, combining reference counting, garbage collection, and memory pools, is efficient and mostly transparent to the programmer, allowing Python developers to focus more on logic and less on memory management. However, it is still important to understand how these mechanisms work to avoid memory-related issues like leaks.

Read More

If you can’t get enough from this article, Aihirely has plenty more related information, such as Python 3.x interview questions, Python 3.x interview experiences, and details about various Python 3.x job positions. [Click here](https://www.aihirely.com/tag/Python 3.x) to check it out.

Related Posts

Trace Job opportunities

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

Get Started Now