オブジェクト指向プログラミング(OOP)におけるカプセル化とデータ隠蔽は、オブジェクトの内部状態を保護し、外部からの不正なアクセスや変更を防ぐための重要な概念です。このセクションでは、Pythonを用いてカプセル化とデータ隠蔽の基本的な考え方と実装方法を初心者にもわかりやすく解説します。
カプセル化は、オブジェクトのデータ(属性)とそれに関連する操作(メソッド)を一つにまとめ、外部からのアクセスを制限する設計原則です。これにより、オブジェクトの内部状態を一貫性のあるものに保ち、システムの複雑さを管理しやすくします。
データ隠蔽は、オブジェクトの内部データを外部から直接アクセスできないようにする手法です。これにより、データの不整合や不正な変更を防ぎます。Pythonでは、アクセス修飾子を使ってデータの可視性を制御します。
Pythonには、主に以下の3種類のアクセス修飾子があります:
_
)で始まる属性で、同じクラスやサブクラスからアクセス可能。__
)で始まる属性で、クラス外からのアクセスを制限。パブリック属性は、どこからでもアクセス可能です。以下の例では、name
とage
がパブリック属性です。
class Person:
def __init__(self, name, age):
self.name = name # パブリック属性
self.age = age # パブリック属性
def greet(self):
print(f"こんにちは、私の名前は{self.name}です。年齢は{self.age}歳です。")
# オブジェクトの生成
person = Person("太郎", 25)
# パブリック属性へのアクセス
print(person.name) # 出力: 太郎
print(person.age) # 出力: 25
# パブリック属性の変更
person.age = 26
print(person.age) # 出力: 26
プロテクテッド属性は、アンダースコア1つ(_
)を属性名の前に付けて定義します。これは、プログラマに対して「この属性は内部で使用するものです。直接アクセスしないでください。」という合図です。
class Employee:
def __init__(self, name, salary):
self.name = name # パブリック属性
self._salary = salary # プロテクテッド属性
def get_salary(self):
return self._salary
def set_salary(self, amount):
if amount > 0:
self._salary = amount
else:
print("給与は正の数でなければなりません。")
# オブジェクトの生成
employee = Employee("花子", 50000)
# プロテクテッド属性へのアクセス(非推奨)
print(employee._salary) # 出力: 50000
# メソッドを通じたアクセス
print(employee.get_salary()) # 出力: 50000
# メソッドを通じた属性の変更
employee.set_salary(55000)
print(employee.get_salary()) # 出力: 55000
employee.set_salary(-1000) # 出力: 給与は正の数でなければなりません。
プライベート属性は、アンダースコア2つ(__
)を属性名の前に付けて定義します。これにより、名前マングリングが行われ、クラス外からの直接アクセスが困難になります。
class BankAccount:
def __init__(self, owner, balance=0):
self.owner = owner # パブリック属性
self.__balance = balance # プライベート属性
def deposit(self, amount):
if amount > 0:
self.__balance += amount
print(f"{amount}円を入金しました。現在の残高は{self.__balance}円です。")
else:
print("入金額は正の数でなければなりません。")
def withdraw(self, amount):
if 0 < amount <= self.__balance:
self.__balance -= amount
print(f"{amount}円を引き出しました。現在の残高は{self.__balance}円です。")
else:
print("引き出し額が不正です。")
def get_balance(self):
return self.__balance
# オブジェクトの生成
account = BankAccount("佐藤", 10000)
# プライベート属性へのアクセス(エラーが発生)
# print(account.__balance) # AttributeError: 'BankAccount' object has no attribute '__balance'
# メソッドを通じたアクセス
print(account.get_balance()) # 出力: 10000
# メソッドを通じた属性の変更
account.deposit(5000) # 出力: 5000円を入金しました。現在の残高は15000円です。
account.withdraw(2000) # 出力: 2000円を引き出しました。現在の残高は13000円です。
print(account.get_balance()) # 出力: 13000
名前マングリング(Name Mangling)とは、Pythonにおいてクラス定義内で特定の名前付け規則に従った属性やメソッド名が、実際には別の内部的な名前へと自動的に書き換えられる仕組みのことを指します。
特に、クラス内部で定義された名前が「ダブルアンダースコアで始まり、ダブルアンダースコアで終わらない」場合に、Pythonはその名前をクラス名で変換した別の名前(マングリング後の名前)として内部管理します。
class MyClass:
def __init__(self):
self.__value = 42
def __method(self):
return "secret method"
obj = MyClass()
print(dir(obj))
上記の場合、__value
や__method
はクラス定義内部では「__value」「__method」と書きますが、実際にはMyClass__value
や`MyClass__methodPといった名前に置き換えられます。
このような変換を名前マングリングと呼びます。
_
)でプロテクテッド属性を、アンダースコア2つ(__
)でプライベート属性を表現します。実際にコードを書いて試すことで、カプセル化とデータ隠蔽の概念とその実装方法をより深く理解しましょう。