6
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

プロパティとは

Pythonには「プロパティ(property)」という、クラスの属性(フィールド)のアクセス方法を
カスタマイズするための仕組みです。

プロパティを使うことで、クラスの内部の実装を隠蔽し、属性へのアクセスや変更を制御できます。
つまり、クラスの変数について考えなくても属性の制御ができるようになるのです。

実際に見てみたほうが分かりやすいと思います。

class Person:
    def __init__(self, name):
        # この属性は「プライベート」にして、直接アクセスしないようにします
        # 変数名の最初に「_」をつけることでプライベートにしています
        self._name = name  

    @property
    def name(self):
        return self._name

    @name.setter
    def name(self, value):
        if not value:
            raise ValueError("名前は空にできません")
        self._name = value

    @name.deleter
    def name(self):
        raise AttributeError("名前は削除できません")

# クラスの使い方
person = Person("Alice")
print(person.name)  # getterが呼ばれる -> "Alice"
person.name = "Bob"  # setterが呼ばれる
print(person.name)  # getterが呼ばれる -> "Bob"
del person.name  # deleterが呼ばれる -> AttributeError

このコードではプロパティは3つの例があり、それぞれ以下のように使うことができます。

取得: person.name は @property デコレータが付いた name メソッドを呼び出します。
設定: person.name = "Bob" は @name.setter デコレータが付いた name メソッドを呼び出します。
削除: del person.name は @name.deleter デコレータが付いた name メソッドを呼び出します。

便利な使い方例紹介

1. データの検証

プロパティのgetter/setter内で、取得・設定する値に対して検証を行うことができます。
例えば、以下のように、負の値が設定されないように制限することができます。

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

  @property
  def age(self):
    return self._age

  @age.setter
  def age(self, value):
    if value < 0:
      raise ValueError("年齢は負の値を設定できません。")
    self._age = value

この例では、ageプロパティのsetter内で、設定値が0未満の場合はエラーを発生させています。
データの整合性を保つ際に便利な使い方だと思います。

2. 計算ロジックのカプセル化

複雑な計算ロジックを、プロパティのgetter/setter内にカプセル化することができます。
例えば、円周計算を行うプロパティを以下のように定義できます。

class Circle:
  def __init__(self, radius):
    self._radius = radius

  @property
  def circumference(self):
    return 2 * math.pi * self._radius

  @circumference.setter
  def circumference(self, value):
    raise NotImplementedError("円周は直接設定できません。")

この例では、circumferenceプロパティのgetter内で円周を計算し、setterでは設定不可としています。
このように、内部ロジックを隠蔽し、インターフェースをシンプルに保つことに使えます。
VScodeとかのサジェストも見やすくなるのでコーディングがしやくなって便利です。

3. ログを出しまくる

デコレータを利用して、すべてのプロパティにログを出力するような仕組みを作ることができます。

def log_property(cls):
  for name, value in cls.__dict__.items():
    if isinstance(value, property):
      @property
      def wrapper(self):
        print(f"プロパティ '{name}' を取得します。")
        return value.fget(self)

      setattr(cls, name, wrapper)
  return cls

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

setterでこれを使えば、値の状態監視もできます。

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

  @property
  def name(self):
    return self._name

  @name.setter
  def name(self, new_name):
    if self._name != new_name:
      print(f"名前が '{self._name}' から '{new_name}' に変更されました。")
    self._name = new_name

4. 仮想プロパティ

存在しない属性に対して、プロパティのようにアクセスできるようにすることができます。

class MyList(list):
  @property
  def length(self):
    return len(self)

MyListクラスは通常のリストのサブクラスとして定義されており、
lengthプロパティを追加することで、リストの要素数にアクセスできるようにしています。

5. 関連プロパティの自動更新

関連するプロパティ間の整合性を保つために、
一方の値を変更すると自動的に他方の値を更新することができます。

class Address:
  def __init__(self, prefecture, city, address, postcode):
    self.prefecture = prefecture
    self.city = city
    self.address = address
    self._postcode = postcode

  @property
  def postcode(self):
    return self._postcode

  @postcode.setter
  def postcode(self, value):
    self._postcode = value
    # 郵便番号変更時に住所を更新する処理
    self.update_address_from_postcode()

  def update_address_from_postcode(self):
    # 実際の処理は省略
    pass

この例では、postcodeプロパティのsetter内で、郵便番号が変更された際に
update_address_from_postcodeメソッドを呼び出し、住所を更新する処理を実行しています。

6
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
6
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?