Inappropriate Intimacy
This code smell occurs when one part of the system knows too much about the internal details of another, leading to tight coupling. When components are too dependent on each other, it becomes difficult to modify or extend the system without breaking other parts.
A good design principle to follow is the Law of Demeter, also known as the “Don’t talk to strangers” rule. It suggests that a module should only interact with its direct dependencies rather than deeply nested objects.
Symptoms
- A class accesses properties of another object’s properties, exposing too much detail.
- Changes in one part of the code require changes in multiple other places.
- Because multiple classes depend on each other’s internal structures, small changes can cause unintended issues.
Example - Violating Law of Demeter
In this example, a SensorSystem
directly accesses the TemperatureSensor
internal attributes, creating tight coupling.
class TemperatureSensor:
def __init__(self, temperature):
self.temperature = temperature # Internal detail exposed
class SensorSystem:
def __init__(self, sensor):
self.sensor = sensor
def get_temperature(self):
# Law of Demeter violation: Directly accessing sensor's attribute
return self.sensor.temperature
# Usage
= TemperatureSensor(25)
sensor = SensorSystem(sensor)
system = system.get_temperature()
temperature print(temperature) # 25
Problem: The SensorSystem
class depends on the internal structure of TemperatureSensor
. If the way temperature is stored changes (e.g., a new sensor model), SensorSystem
must also change.
Solutions
Example solution - Using getter methods
Instead of directly accessing attributes, define getter methods in TemperatureSensor
to limit exposure.
class TemperatureSensor:
def __init__(self, temperature):
self._temperature = temperature # Use a private variable
def get_temperature(self):
return self._temperature # Encapsulated access
class SensorSystem:
def __init__(self, sensor):
self.sensor = sensor
def get_temperature(self):
return self.sensor.get_temperature() # Indirect access through method
# Usage
= TemperatureSensor(25)
sensor = SensorSystem(sensor)
system print(system.get_temperature()) # 25
Why is this better?
- The
SensorSystem
no longer needs to know the internal structure ofTemperatureSensor
. - If
TemperatureSensor
changes, onlyget_temperature()
needs to be updated, not every place it’s used.
Example solution - Removing the dependency
A better design is to pass only the needed data instead of an entire object.
class SensorSystem:
def __init__(self, temperature):
self.temperature = temperature
def get_temperature(self):
return self.temperature # Works directly with the value
# Usage
= 25
temperature = SensorSystem(temperature)
system print(system.get_temperature()) # 25
Why is this better?
SensorSystem
no longer depends onTemperatureSensor
, making it more modular and reusable.- Works even if the source of temperature data changes (e.g., from a file, API, or another sensor).
Balance between dependecy injection and encapsulation. If the data is simple and does not require complex operations, pass it directly. If the data is complex or requires additional logic, encapsulate it in a class.
Key Takeaways
- Follow the Law of Demeter - Only interact with direct dependencies.
- Encapsulate data - Use getters and setters to access and modify data.
- Reduce dependencies - pass only the necessary information to other components.