Descriptors

Python Descriptors

Python Descriptors are a way to customize the access, assignment and deletion of object attributes. They provide a powerful mechanism for managing the behavior of attributes by defining methods that get, set and delete their values. Descriptors are often used to implement properties, methods and attribute validation.

descriptor is any object that implements at least one of the methods such as __get__, __set__ and __delete__. These methods control how an attribute’s value is accessed and modified.

How Python Descriptors Work?

When an attribute is accessed on an instance then Python looks up the attribute in the instance’s class. If the attribute is found and it is a descriptor then Python invokes the appropriate descriptor method instead of simply returning the attribute’s value. This allows the descriptor to control what happens during attribute access.

The descriptor protocol is a low-level mechanism that is used by many high-level features in Python such as properties, methods, static methods and class methods. Descriptors can be used to implement patterns like lazy loading, type checking and computed properties.

Descriptor Methods

Python Descriptors involve three main methods namely __get__(), __set__() and __delete__(). As we already discussed above these methods control the behavior of attribute access, assignment and deletion, respectively.

1. The __get__() Method

The __get__() method in descriptors is a key part of the descriptor protocol in Python. It is called to retrieve the value of an attribute from an instance or from the class. Understanding how the __get__() method works is crucial for creating custom descriptors that can manage attribute access in sophisticated ways.

Syntax

The following is the syntax of Python Descriptor __get__ method −

def__get__(self, instance, owner):"""
   instance: the instance that the attribute is accessed through, or None when accessed through the owner class.
   owner: the owner class where the descriptor is defined.
   """

Parameters

Below are the parameters of this method −

  • self: The descriptor instance.
  • instance: The instance of the class where the attribute is accessed. It is None when the attribute is accessed through the class rather than an instance.
  • owner: The class that owns the descriptor.

Example

Following is the basic example of __get__() method in which it returns the stored value _value when obj.attr is accessed −

classDescriptor:def__get__(self, instance, owner):if instance isNone:return self
  return instance._value
classMyClass: attr = Descriptor()def__init__(self, value):
  self._value = value
obj = MyClass(42)print(obj.attr)

Output

42

2. The __set__() Method

The __set__() method is part of the descriptor protocol in Python and is used to control the behavior of setting an attribute’s value. When an attribute managed by a descriptor is assigned a new value then the __set__() method is called by allowing the user to customize or enforce rules for the assignment.

Syntax

The following is the syntax of Python Descriptor __set__() method −

def__set__(self, instance, value):"""
instance: the instance of the class where the attribute is being set.
value: the value to assign to the attribute.
"""</pre>

Parameters

Below are the parameters of this method −

  • self: The descriptor instance.
  • instance: The instance of the class where the attribute is being set.
  • value: The value being assigned to the attribute.

Example

Following is the basic example of __set__() method in which ensures that the value assigned to attr is an integer −

classDescriptor:def__set__(self, instance, value):ifnotisinstance(value,int):raise TypeError("Value must be an integer")
    instance._value = value
classMyClass:
attr = Descriptor()def__init__(self, value):
    self.attr = value
obj = MyClass(42)print(obj.attr) obj.attr =100print(obj.attr)

Output

<__main__.Descriptor object at 0x000001E5423ED3D0>
<__main__.Descriptor object at 0x000001E5423ED3D0>

3. The __delete__() Method

The __delete__() method in the descriptor protocol allows us to control what happens when an attribute is deleted from an instance. This can be useful for managing resources, cleaning up or enforcing constraints when an attribute is removed.

Syntax

The following is the syntax of Python Descriptor __delete__() method −

def__delete__(self, instance):"""
instance: the instance of the class from which the attribute is being deleted.
"""</pre>

Parameters

Below are the parameters of this method −

  • self: The descriptor instance.
  • instance: The instance of the class where the attribute is being deleted.

Example

Following is the basic example of __set__() method in which ensures that the value assigned to attr is an integer −

classLoggedDescriptor:def__init__(self, name):
  self.name = name
def__get__(self, instance, owner):return instance.__dict__.get(self.name)def__set__(self, instance, value):
  instance.__dict__[self.name]= value
def__delete__(self, instance):if self.name in instance.__dict__:print(f"Deleting {self.name} from {instance}")del instance.__dict__[self.name]else:raise AttributeError(f"{self.name} not found")classPerson: name = LoggedDescriptor("name") age = LoggedDescriptor("age")def__init__(self, name, age):
  self.name = name
  self.age = age
# Example usage p = Person("Tutorialspoint",30)print(p.name)print(p.age)del p.name print(p.name)del p.age print(p.age)

Output

Tutorialspoint
30
Deleting name from <__main__.Person object at 0x0000021A1A67E2D0>
None
Deleting age from <__main__.Person object at 0x0000021A1A67E2D0>
None

Types of Python Descriptors

In Python descriptors can be broadly categorized into two types based on the methods they implement. They are −

  • Data Descriptors
  • Non-data Descriptors

Let's see about the two types of python descriptors in detail for our better understanding.

1. Data Descriptors

Data descriptors are a type of descriptor in Python that define both __get__()and __set__() methods. These descriptors have precedence over instance attributes which meand that the descriptors __get__()and __set__() methods are always called, even if an instance attribute with the same name exists.

Example

Below is the example of a data descriptor that ensures an attribute is always an integer and logs access and modification operations −

classInteger:def__get__(self, instance, owner):print("Getting value")return instance._value

   def__set__(self, instance, value):print("Setting value")ifnotisinstance(value,int):raise TypeError("Value must be an integer")
  instance._value = value
def__delete__(self, instance):print("Deleting value")del instance._value classMyClass: attr = Integer()# Usage obj = MyClass() obj.attr =42print(obj.attr) obj.attr =100print(obj.attr)del obj.attr

Output

Setting value
Getting value
42
Setting value
Getting value
100
Deleting value

2. Non-data Descriptors

Non-data descriptors are a type of descriptor in Python that define only the __get__() method. Unlike data descriptors, non-data descriptors can be overridden by instance attributes. This means that if an instance attribute with the same name exists then it will take precedence over the non-data descriptor.

Example

Following is an example of a non-data descriptor that provides a default value if the attribute is not set on the instance −

classDefault:def__init__(self, default):
  self.default = default
def__get__(self, instance, owner):returngetattr(instance,'_value', self.default)classMyClass: attr = Default("default_value")# Usage obj = MyClass()print(obj.attr) obj._value ="Tutorialspoint"print(obj.attr)

Output

default_value
Tutorialspoint

Data Descriptors Vs. Non-data Descriptors

Understanding the differences between Data Descriptors and Non-data Descriptors of python Descriptors is crucial for leveraging their capabilities effectively.

CriteriaData DescriptorsNon-Data Descriptors
DefinitionImplements both __get__(), __set__() methods, and the __delete__() method optionally.Implements only __get__() method.
Methods__get__(self, instance, owner)
__set__(self, instance, value)
__delete__(self, instance) (optional)
__get__(self, instance, owner)
PrecedenceTakes precedence over instance attributes.Overridden by instance attributes.
Use CasesAttribute validation and enforcement,
Managed attributes (e.g., properties),
Logging attribute access and modification,
Enforcing read-only attributes.
Method binding,
Caching and,
Providing default values..

Finally we can say Descriptors in Python provide a powerful mechanism for managing attribute access and modification. Understanding the differences between data descriptors and non-data descriptors as well as their appropriate use cases is essential for creating robust and maintainable Python code.

By leveraging the descriptor protocol developers can implement advanced behaviors such as type checking, caching and read-only properties.

Comments

Leave a Reply

Your email address will not be published. Required fields are marked *