I recently stumbled upon a code from one of my coworker who is rather new to Python. He is coming from a Java/Scala background and used python's property
decorator to mimic Java Getters/Setters. Let's see why you should not do this.
class ParquetReader:
def __init__(self, spark_session):
self._spark_session = spark_session
@property
def spark_session(self):
return self._spark_session
This made me think about this article which outline pretty well the reason why you should not do this in Python.
The TL;DR is: "It is not Pythonic". Don't blindly apply what you've seen in other languages like Java directly into Python.
Avoid exposing your objects internals
First, you should not expose your object internals except if it is a data class (i.e: A structure for holding pure data without behavior; by the way, you should not create objects mixing both data and behavior).
The reason is that it may cause client code to use those internals, creating strong coupling between each other and greatly reducing your ability to refactor your code. If that case arise (using another object's internals), it might be the sign of a Feature Envy code smell. You can try the Tell Don't Ask principle as well to help you mitigate this.
Use property
for dynamic computations on the fly
The main purpose of the property
decorator is to allow you to perform additional computations when reading/writing/deleting an attribute. If there is no additional behavior, just stick to the raw attribute directly.
# Details for observer object is omitted
class Foo:
def __init__(self, observer):
self._bar = ""
self.observer = observer
@property
def bar(self):
return self._bar
@bar.setter
def bar(self, value):
self._bar = value
self.observer.notify(f"bar updated: {self.bar}")
Use property
for refactoring without breaking changes
Another reason to use that decorator is if you plan on changing the attribute name without introducing breaking change.
class ParquetReader:
def __init__(self, spark_session):
self._session = spark_session
@property
def spark_session(self):
return self._session
Here we can safely rename our internal attribute without changing the one exposed.
Do not use property
for aesthetic reasons
You should not use the property
decorator to turns a real function into a simple attribute just because it looks nicer in particular if this function is doing some heavy computation (or at least, non neglictible one). Otherwise, clients might be tempted to use repeatedly this "attribute" causing performances issues
# Bad
class Circle:
@property
def area(self) -> float:
...
# Good
class Circle:
def area(self) -> float:
...