Better way 44 ์ธํฐ์ ๊ฒํฐ ๋ฉ์๋ ๋์ ํ๋ฒํ ์ ํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ฌ์ฉํ๋ผ
setter, getter ๋ฉ์๋ ํ์ฉ์ด ์๋ ์ ํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ง์ ํ์ฉํ์
# ์ ์ข์ ์์ - getter, setter ํ์ฉ
class Person:
def __init__(self):
self._name = ''
def get_name(self):
return self._name
def set_name(self, value):
self._name = value
p = Person()
p.set_name('Sungchan')
print(p.get_name())
# ์ข์ ์์ - ์ ํธ๋ฆฌ๋ทฐํธ ํ์ฉ
class Person:
def __init__(self):
self.name = ''
p = Person()
p.name = 'Sungchan'
print(p.name)
๋ง์ฝ์ ํน์ ์ ํธ๋ฆฌ๋ทฐํธ๊ฐ ์ค์ ๋ ๋, ํน๋ณํ ๊ธฐ๋ฅ์ ์ํํ๋ค๋ฉด @property๋ฅผ ํ์ฉํ์
class Person:
def __init__(self):
self._age = 0
@property # ์ ํธ๋ฆฌ๋ทฐํธ ๊ฐ ์กฐํ ์, ํด๋น ํจ์ ํธ์ถ
def age(self):
return self._age
@age.setter # age ์ ํธ๋ฆฌ๋ทฐํธ ๊ฐ ๋ณ๊ฒฝ ์, ํด๋น ํจ์ ํธ์ถ
def age(self, value):
if value < 0:
raise ValueError("๋์ด๋ ์์๊ฐ ๋ ์ ์์ต๋๋ค.")
self._age = value
p = Person()
p.age = 25
print(p.age) # 25
p.age = -3 # ValueError: ๋์ด๋ ์์๊ฐ ๋ ์ ์์ต๋๋ค.
@property ์ฌ์ฉ์ ์ฃผ์์
- I/O ๋๋ DB ์กฐํ์ ๊ฐ์ ์ํ ์๊ฐ์ด ์ค๋ ๊ฑธ๋ฆฌ๊ฑฐ๋ ๋น์ฉ์ด ๋น์ผ ๋ด์ฉ์ ์ฌ์ฉํ์ง ์๋๋ค.
- ๋ค๋ฅธ ๊ฐ์ฒด๋ฅผ ์กฐํํ๊ฑฐ๋ ํฌํจํ๋ ๋ณต์กํ ์ฝ๋๋ฅผ ์์ฑํ์ง ์๋๋ค.
-> ๊ฐ๋ฐ์๊ฐ ์ธ์ํ๊ธฐ ์ด๋ ค์, ์ผ๋ฐ ๋ฉ์๋๋ฅผ ์ฌ์ฉํ์
Better way 45 ์ ํธ๋ฆฌ๋ทฐํธ๋ฅผ ๋ฆฌํฉํฐ๋งํ๋ ๋์ @property๋ฅผ ์ฌ์ฉํ๋ผ
์ํ์ ๊ฐ๊ฒฉ๊ณผ ํ ์ธ ์ ์ฑ ์ ๊ด๋ฆฌํ๋ ๊ฐ์ฒด๊ฐ ์๋ค.
๊ฐ๊ฒฉ๊ณผ ํ ์ธ ์ ์ฑ ์ด ํญ์ ์ฐ๋๋๋ ๊ฐ์ฒด๊ฐ ํ์ํ๋ค๊ณ ๊ฐ์ ํ์.
class Product:
def __init__(self, price):
self._price = price
self._discount_rate = 0.1 # ๊ธฐ๋ณธ ํ ์ธ์จ 10%
@property
def price(self):
return int(self._price * (1 - self._discount_rate))
@price.setter
def price(self, value):
if value < 0:
raise ValueError("๊ฐ๊ฒฉ์ ์์๊ฐ ๋ ์ ์์ต๋๋ค.")
self._price = value
@property
def discount_rate(self):
return self._discount_rate
@discount_rate.setter
def discount_rate(self, value):
if not 0 <= value <= 1:
raise ValueError("ํ ์ธ์จ์ 0~1 ์ฌ์ด์ฌ์ผ ํฉ๋๋ค.")
self._discount_rate = value
# ์ฌ์ฉ ์์
p = Product(10000)
print(p.price) # 9000 (10% ํ ์ธ)
p.discount_rate = 0.2
print(p.price) # 8000 (20% ํ ์ธ)
p.price = 20000
print(p.price) # 16000 (20% ํ ์ธ ์ ์ฉ๋ ์ ๊ฐ๊ฒฉ)
Better way 46 ์ฌ์ฌ์ฉ ๊ฐ๋ฅํ @property ๋ฉ์๋๋ฅผ ๋ง๋ค๋ ค๋ฉด ๋์คํฌ๋ฆฝํฐ๋ฅผ ์ฌ์ฉํ๋ผ
@propery์ ๋จ์ ์ผ๋ก๋ ์ฌ์ฌ์ฉ์ฑ์ด ์ด๋ ต๋ค.
๊ฐ๊ฐ์ property ๋ง๋ค @propery, @property.setter ์ ํด๋นํ๋ ๋ฉ์๋๋ฅผ ๊ตฌํํด์ผํ๋ค.
๋์ผํ ๋ก์ง, ๋ด์ฉ์ property๋ฅผ ํ์ฉํ ๋๋ ๋์คํฌ๋ฆฝํฐ๋ฅผ ์ฌ์ฉํ์.
# ์ ์ข์ ์์ - propery ๊ฐ๊ฐ ๊ตฌํ
class Product:
def __init__(self, price, weight):
self._price = price
self._weight = weight
@property
def price(self):
return self._price
@price.setter
def price(self, value):
if value < 0:
raise ValueError("๊ฐ๊ฒฉ์ ์์์ผ ์ ์์ต๋๋ค.")
self._price = value
@property
def weight(self): # price property ์ ๋ก์ง ์ค๋ณต
return self._weight
@weight.setter
def weight(self, value): # price property ์ ๊ฒ์ฆ ๋ก์ง ์ค๋ณต
if value < 0:
raise ValueError("๋ฌด๊ฒ๋ ์์์ผ ์ ์์ต๋๋ค.")
self._weight = value
class Positive: # ๋์คํฌ๋ฆฝํฐ
def __set_name__(self, owner, name):
self.name = f"_{name}"
def __get__(self, instance, owner): # ๋์คํฌ๋ฆฝํฐ๋ก ์ฌ์ฉํ๊ธฐ ์ํด __get__, __set__ ๊ตฌํ ํ์
return getattr(instance, self.name)
def __set__(self, instance, value): # ๋์คํฌ๋ฆฝํฐ๋ก ์ฌ์ฉํ๊ธฐ ์ํด __get__, __set__ ๊ตฌํ ํ์
if value < 0:
raise ValueError(f"{self.name[1:]}๋(์) ์์์ผ ์ ์์ต๋๋ค.")
setattr(instance, self.name, value)
class Product:
price = Positive() # Positive ๋์คํฌ๋ฆฝํฐ ํ์ฉ
weight = Positive() # Positive ๋์คํฌ๋ฆฝํฐ ํ์ฉ
def __init__(self, price, weight):
self.price = price
self.weight = weight
p = Product(10000, 2.5)
print(p.price) # 10000
print(p.weight) # 2.5
p.price = 5000 # ์ ์
# p.weight = -1 # ValueError: weight๋(์) ์์์ผ ์ ์์ต๋๋ค.
Better way 47 ์ง์ฐ ๊ณ์ฐ ์ ํธ๋ฆฌ๋ทฐํธ๊ฐ ํ์ํ๋ฉด __getattr__, __getattribute__, __setattr__์ ์ฌ์ฉํ๋ผ
๋ฉ์๋ | ํธ์ถ ์์ | ์ฉ๋ |
__getattribute__ | ๋ชจ๋ ์์ฑ ์ ๊ทผ ์ ๋ฌด์กฐ๊ฑด ํธ์ถ | ์์ฑ ์ ๊ทผ ์ ์ฒ๋ฆฌ, ๊ฐ์, ๋ก๊น ๋ฑ |
__getattr__ | ์์ฑ์ด ์กด์ฌํ์ง ์์ ๋๋ง ํธ์ถ | ๋์ ์์ฑ ์์ฑ, ์ง์ฐ ๊ณ์ฐ ๋ฑ |
__setattr__ | ๋ชจ๋ ์์ฑ ์ค์ ์ ํธ์ถ | ๊ฐ ๊ฒ์ฆ, ์๋ ๋ก๊ทธ, ๋ด๋ถ ์ ์ฅ ๋ฐฉ์ ๋ณ๊ฒฝ ๋ฑ |
class DynamicAttrs: # ์ ํธ๋ฆฌ๋ทฐํธ๊ฐ ์์ผ๋ฉด ๋์ ์ผ๋ก ์ ํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ถ๊ฐํ๋ ํด๋์ค
def __init__(self):
self.existing = 42
def __getattribute__(self, name): # ์ ํธ๋ฆฌ๋ทฐํธ ์ ๊ทผ ์ - ๋ฌด์กฐ๊ฑด ํธ์ถ
print(f"[getattribute] {name} ์ ๊ทผ ์๋")
return super().__getattribute__(name) # ๋ฌดํ ์ฌ๊ท ๋ฐฉ์ง!
def __getattr__(self, name): # ์ ํธ๋ฆฌ๋ทฐํธ๊ฐ ์กด์ฌํ์ง ์์ ๋
print(f"[getattr] {name} ์์ฑ์ด ์กด์ฌํ์ง ์์ ๋์ ์ผ๋ก ์์ฑ")
value = f"{name}์ ์ํ ๊ฐ" # ์์ ๋ณ์ ๊ฐ ์์ฑ
setattr(self, name, value) # ์ ํธ๋ฆฌ๋ทฐํธ ์ถ๊ฐ
return value
def __setattr__(self, name, value): # ์ ํธ๋ฆฌ๋ทฐํธ ๊ฐ ์์ ์ ํธ์ถ
print(f"[setattr] {name} ์ค์ : {value}")
super().__setattr__(name, value)
obj = DynamicAttrs()
# ์กด์ฌํ๋ ์์ฑ ์ ๊ทผ
print(obj.existing)
# => [getattribute] existing ์ ๊ทผ ์๋
# => 42
# ์กด์ฌํ์ง ์๋ ์์ฑ ์ ๊ทผ โ getattr ์๋
print(obj.missing)
# => [getattribute] missing ์ ๊ทผ ์๋
# => [getattr] missing ์์ฑ์ด ์กด์ฌํ์ง ์์ ๋์ ์ผ๋ก ์์ฑ
# => missing์ ์ํ ๊ฐ
# ๋ค์ ์ ๊ทผ ์ โ ์ด๋ฏธ ์กด์ฌํ๋ฏ๋ก getattr์ ํธ์ถ๋์ง ์์
print(obj.missing)
# => [getattribute] missing ์ ๊ทผ ์๋
# => missing์ ์ํ ๊ฐ
Better way 48 __init_subclass__๋ฅผ ์ฌ์ฉํด ํ์ ํด๋์ค๋ฅผ ๊ฒ์ฆํ๋ผ
์์๋ฐ๋ ํ์ ํด๋์ค๊ฐ ํน์ ๊ท์น์ ๋ฐ๋ฅด๋๋ก ๊ฐ์ ํ๊ณ ์ถ์ ๋, __init_subclass__๋ฅผ ํ์ฉํ ์ ์๋ค.
์ถ์ ๋ฉ์๋๋ฅผ ์ ์ธํ๊ณ ํ์ ํด๋์ค์์ ๊ตฌํํ๋ ๊ฒ๋ณด๋ค ์ ์ฐํ๋ค๋ ์ฅ์ ์ด ์๋ค.
๋ฏน์ค์ธ ํ์ฌ ์ฌ์ฉํ๊ธฐ๋ ํจ์ฌ ์ข๋ค.
๋ ๊ฐ์ง ๋ถ๋ชจ ํด๋์ค๊ฐ ์๋ค.
1. ๋ค๊ฐํ์ด ์ ํจํ์ง ๊ฒ์ฌ (sides >= 3)
2. ์์ด ํ๋์์ ์ผ๋์ง ๊ฒ์ฌ (color == 'blue')
class Polygon: # sides ๊ฐ 3๊ฐ ์ด์์ธ์ง ํ์ธ
def __init_subclass__(cls):
super().__init_subclass__()
if not hasattr(cls, 'sides'):
raise TypeError(f"{cls.__name__}๋ 'sides' ์์ฑ์ ์ ์ํด์ผ ํฉ๋๋ค.")
if cls.sides < 3:
raise ValueError(f"{cls.__name__}๋ ์ต์ 3๊ฐ์ ๋ณ์ ๊ฐ์ ธ์ผ ํฉ๋๋ค.")
class Triangle(Polygon):
sides = 3 # โ
์ ํจ
class Line(Polygon):
sides = 1 # โ ValueError: Line๋ ์ต์ 3๊ฐ์ ๋ณ์ ๊ฐ์ ธ์ผ ํฉ๋๋ค.
๋ฏน์ค์ธ ์์
class BlueShape:
def __init_subclass__(cls):
super().__init_subclass__()
if not hasattr(cls, 'color'):
raise TypeError(f"{cls.__name__}๋ 'color' ์์ฑ์ ์ ์ํด์ผ ํฉ๋๋ค.")
if cls.color != 'blue':
raise ValueError(f"{cls.__name__}๋ color๊ฐ ๋ฐ๋์ 'blue'์ฌ์ผ ํฉ๋๋ค.")
class BlueSquare(Polygon, BlueShape): # ๋ฏน์ค์ธํ์ฌ ๋ ๊ฐ์ง ํด๋์ค ๊ฒ์ฆ ๊ฐ๋ฅ
sides = 4
color = 'blue' # โ
์ ํจ
class RedSquare(Polygon, BlueShape): # ๋ฏน์ค์ธํ์ฌ ๋ ๊ฐ์ง ํด๋์ค ๊ฒ์ฆ ๊ฐ๋ฅ
sides = 4
color = 'red' # โ ValueError: RedSquare๋ color๊ฐ ๋ฐ๋์ 'blue'์ฌ์ผ ํฉ๋๋ค.
Better way 49 __init_subclass__๋ฅผ ์ฌ์ฉํด ํด๋์ค ํ์ฅ์ ๋ฑ๋กํ๋ผ
ํ๋ฌ๊ทธ์ธ, ๋ช ๋ น์ด, ํธ๋ค๋ฌ ๋ฑ์ ๋ง๋ค ๋, ์์์ผ๋ก ํ๋ฉด ์๋์ผ๋ก ๋ฑ๋ก๋๋ ๊ตฌ์กฐ๋ฅผ ๊ตฌํํ ์ ์๋ค.
class CommandBase: # ๋ช
๋ น์ด ์งํฉ
registry = {}
def __init_subclass__(cls, **kwargs):
super().__init_subclass__(**kwargs)
name = cls.__name__.lower().replace("command", "")
CommandBase.registry[name] = cls
class HelloCommand(CommandBase): # CommandBase๋ฅผ ์์ -> CommandBase.registry์ ๋ฑ๋ก
def run(self):
print("์๋
ํ์ธ์!")
class ExitCommand(CommandBase): # CommandBase๋ฅผ ์์ -> CommandBase.registry์ ๋ฑ๋ก
def run(self):
print("์ข
๋ฃํฉ๋๋ค.")
# ๋ฑ๋ก๋ ์ปค๋งจ๋ ํ์ธ
print(CommandBase.registry)
# {'hello': <class '__main__.HelloCommand'>, 'exit': <class '__main__.ExitCommand'>}
# ์ปค๋งจ๋ ์คํ
CommandBase.registry['hello']().run() # ์๋
ํ์ธ์!
CommandBase.registry['exit']().run() # ์ข
๋ฃํฉ๋๋ค.
Better way 51 ํฉ์ฑ ๊ฐ๋ฅํ ํด๋์ค ํ์ฅ์ด ํ์ํ๋ฉด ๋ฉํํด๋์ค๋ณด๋ค๋ ํด๋์ค ๋ฐ์ฝ๋ ์ดํฐ๋ฅผ ์ฌ์ฉํ๋ผ
๋น๊ต ํญ๋ชฉ | ํด๋์ค ๋ฐ์ฝ๋ ์ดํฐ (Python) | ๋ฉํํด๋์ค (Python) | ์ด๋ ธํ ์ด์ (Java) |
์ ์ฉ ๋์ | ํด๋์ค ์ ์ฒด | ํด๋์ค ์ ์ฒด | ํด๋์ค, ๋ฉ์๋, ํ๋ ๋ฑ |
์ฌ์ฉ ๋ฐฉ์ | @decorator | metaclass=... | @AnnotationName |
์ฌ์ฉ ์์ | ํด๋์ค ์ ์ ํ - ํด๋์ค ์ ์ฒด๊ฐ ๋ง๋ค์ด์ง ๋ค์ @decorator(cls) ํธ์ถ |
ํด๋์ค ์ ์ ์ค - type.__new__ ์ด๋ __init__ ์์ ํด๋์ค ์์ฑ ๊ณผ์ ์์ ๊ด์ฌ |
์ปดํ์ผ ์ / ๋ฐํ์ ์ |
๊ธฐ๋ฅ ๋ณ๊ฒฝ ๋ฐฉ์ | ๋ฉ์๋ ์ถ๊ฐ, ์์ฑ ๋ณ๊ฒฝ ๋ฑ ๋์ ํ์ฅ | ํด๋์ค ์์ฑ ๋ฐฉ์ ๋ณ๊ฒฝ | ๋ฆฌํ๋ ์ ๊ธฐ๋ฐ ์ฒ๋ฆฌ (ํ๋ ์์ํฌ๊ฐ ํด์ํจ) |
ํฉ์ฑ ๊ฐ๋ฅ์ฑ | O (์ฌ๋ฌ ๊ฐ ์ค์ฒฉ ๊ฐ๋ฅ) | X (ํ๋๋ง ์ฌ์ฉ ๊ฐ๋ฅ) | O (์ฌ๋ฌ ์ด๋ ธํ ์ด์ ๊ฐ๋ฅ) |
ํด๋์ค ๋ด ๋ฉ์๋์ ๋์์ ๋ก๊น ํ๋ ํด๋์ค ๋ฐ์ฝ๋ ์ดํฐ
def enhance_class(cls): # ํด๋์ค ๋ฐ์ฝ๋ ์ดํฐ, ๋ฉ์๋ ๋์์ ๋ก๊น
ํจ
# ๋ชจ๋ ๋ฉ์๋์ ๋ก๊น
์ถ๊ฐ
for attr_name, attr_value in cls.__dict__.items():
if callable(attr_value):
def wrapper(func):
def wrapped(self, *args, **kwargs):
print(f"[LOG] {func.__name__} called")
return func(self, *args, **kwargs)
return wrapped
setattr(cls, attr_name, wrapper(attr_value))
# ํด๋์ค์ ๊ณตํต ์์ฑ ์ถ๊ฐ
cls.created_by = "system"
return cls
@enhance_class
class User:
def __init__(self, name):
self.name = name
def greet(self):
print(f"Hello, {self.name}")
u = User("pasa")
u.greet()
print(u.created_by)
# ์ถ๋ ฅ
# [LOG] __init__ called
# [LOG] greet called
# Hello, pasa
# system
์ ๋ฆฌ
๋น๊ต | ํด๋์ค ๋ฐ์ฝ๋ ์ดํฐ | ๋ฉํํด๋์ค |
ํธ์ถ ์์ | ํด๋์ค ์ ์ ํ | ํด๋์ค ์ ์ ์ค |
๊ฐ์ ๋์ | ์์ฑ๋ ํด๋์ค ๊ฐ์ฒด | ํด๋์ค ์์ฑ ํ๋ก์ธ์ค |
์ ์ฐ์ฑ | ๊ฐ๋จํ ํ์ฅ, ํฉ์ฑ์ ์ ๋ฆฌ | ๊ตฌ์กฐ์ ์ธ ์ ์ด์ ๊ฐ๋ ฅ |
๋์ด๋ | ๋ฎ์ | ๋์ |
'๐ Python' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
์ดํํฐ๋ธ ํ์ด์ฌ 7์ฅ - ๋์์ฑ๊ณผ ๋ณ๋ ฌ์ฑ (0) | 2025.04.01 |
---|---|
์ดํํฐ๋ธ ํ์ด์ฌ 5์ฅ - ํด๋์ค์ ์ธํฐํ์ด์ค (0) | 2025.03.21 |
์ดํํฐ๋ธ ํ์ด์ฌ 4์ฅ - ์ปดํ๋ฆฌํจ์ ๊ณผ ์ ๋๋ ์ดํฐ (0) | 2025.03.18 |
์ดํํฐ๋ธ ํ์ด์ฌ 3์ฅ - ํจ์ (0) | 2025.03.17 |
์ดํํฐ๋ธ ํ์ด์ฌ 2์ฅ - ๋ฆฌ์คํธ์ ๋์ ๋๋ฆฌ (0) | 2025.03.12 |