前言
此篇作為python class/OOP 的筆記,因為我一直不清楚python class的操作方法,因此紀錄下來以便日後查找。
主要內容為以下Youtube影片與Github的內容, 影片作者為Corey Schafer
- Youtube: Python OOP Tutorials — Working with Classes
- Github: CoreyMSchafer / code_snippets
1. Classes and Instances
在class中:
data叫作attribute
function叫作method
instance指用class去創造一個物件,也指被class狀造的物件
Instance
從以下的程式碼中可以看到,如果單純建立空的Class Employee,並使用class 去instance一個object。這個object可以直接在class外部用object名稱去建立object裏面的值。
pythonclass Employee: pass # 使用pass避免error emp_1 = Employee() emp_2 = Employee() emp_1.first = "Tiny" emp_1.last = "Murky" emp_1.email = "tiny.murky@company.com" emp_1.pay = 50000 print(emp_1.email) # 輸出: tiny.murky@company.com
但是上述的方法很麻煩,所以可以使用class建立function __init__幫我們初始化class,用法等同於其他語言的constructor。
需要注意的是class中的function第一個argument一定要放self(變數名稱可以自己取,但是儘量是self)。 self的意思是指instance自己,舉例來說如果建立emp_1 = Employee()時,init__的self就會是emp_1,也就是指__init(emp_1)的意思。
以下程式碼可以看出 Employee.init(emp_1) 和 emp_1.init()是相同的產出。但是Employee.init()卻回傳少一個值self,這個self就是指instance,例如emp_1。
pythonclass Employee: def __init__(self): pass emp_1 = Employee() print(Employee.__init__(emp_1)) # 輸出: None print(emp_1.__init__()) # 輸出: None print(Employee.__init__()) # 輸出: TypeError: __init__() missing 1 required positional argument: 'self'
有了__init__()之後就可以如下方的程式碼,直接在instance class時把argument放進__init__中,由於self就是instance,效果等同於從class外面一個一個建立instance的attribute。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first # 同等emp_1.first = "Tiny" self.last = last # 同等emp_1.last = "Murky" self.pay = pay # 同等emp_1.pay = 50000 self.email = f"{self.first}{self.last}@company.com" first = "Tiny" last = "Murky" pay = 50000 emp_1 = Employee(first, last, pay) print(emp_1.email) # 輸出: TinyMurky@company.com
Method
如果要在class中創method,記得在argument中放入self,來接取instance自己,如果僅有fullname()這樣子的話會出現訊息:TypeError: fullname() takes 0 positional arguments but 1 was given,代表沒有argument去接收instance自己。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def fullname(self): return f"{self.first} {self.last}" print(emp_1.fullname()) # 輸出 Tiny Murky print(Employee.fullname(emp_1))# 輸出 Tiny Murky
2. Class Variables
如果想要建立一個所有Instance共用的值,可以直接寫在整個class的最上方,instance要使用共用值時可以直接用self取出
pythonclass Employee: raise_amt = 1.05 # 共用值放最上面 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def apply_raise(self): self.pay = int(self.pay * self.raise_amt) # 使用self.raise_amt取出值
從以下程式碼中可以看出,如果從Class中直接更改raise_amt,其他instance也會一起被更改。
pythonprint(emp_1.pay) # 輸出 10,000 emp_1.apply_raise() print(emp_1.pay) # 輸出 10,500 Employee.raise_amt = 1.1 emp_1.pay = 10000 print(emp_1.pay) # 輸出 10,000 emp_1.apply_raise() print(emp_1.pay) # 輸出 11,000
但如果是在instance單獨更改的話,則只會影響到該instance
pythonemp_1 = Employee("Tiny", "Murky", 10000) emp_2 = Employee("Test", "User", 10000) print(emp_1.pay) # 輸出 10,000 emp_1.apply_raise() print(emp_1.pay) # 輸出 10,500 emp_1.raise_amt = 1.1 print(emp_2.pay) # 輸出 10,000 emp_2.apply_raise() print(emp_2.pay) # 輸出 10,500
原因可以從__dict__中觀察,在更改emp_1=raise_amt之前,可以觀察到emp_1內部沒有raise_amt,他是取用Class Employee的raise_amt 1.05。在emp_1= raise_amt之後可以看見emp_1內部出現自己的raise_amt 1.1,優先使用自己的raise_amt,此外Class Employee的raise_amt還是保持在1.05
pythonemp_1 = Employee("Tiny", "Murky", 10000) print(emp_1.__dict__) # {'first': 'Tiny', 'last': 'Murky', 'pay': 10500, 'email': 'TinyMurky@company.com'} print(Employee.__dict__) # {'__module__': '__main__', 'raise_amt': 1.05, '__init__': <function Employee.__init__ at 0x7f5f6920c280>, 'fullname': <function Employee.fullname # at 0x7f5f6920c310>, 'apply_raise': <function Employee.apply_raise at 0x7f5f6920c3a0>, '__dict__': <attribute '__dict__' of 'Employee' objects>, '_ # _weakref__': <attribute '__weakref__' of 'Employee' objects>, '__doc__': None} emp_1.raise_amt = 1.1 print(emp_1.__dict__) # {'first': 'Tiny', 'last': 'Murky', 'pay': 10500, 'email': 'TinyMurky@company.com', # 'raise_amt': 1.1} print(Employee.__dict__) # {'__module__': '__main__', 'raise_amt': 1.05, '__init__': <function Employee.__init__ at 0x7f5f6920c280>, 'fullname': <function Employee.fullname # at 0x7f5f6920c310>, 'apply_raise': <function Employee.apply_raise at 0x7f5f6920c3a0>, '__dict__': <attribute '__dict__' of 'Employee' objects>, '_ # _weakref__': <attribute '__weakref__' of 'Employee' objects>, '__doc__': None}
另外還有一個神奇的用法,可以使用className.attribute的方法在class裏面控制attribute,每instance一次物件,attribute的數值都會變更,但是attribute又在各instance中保持一致。
pythonclass Employee: raise_amt = 1.05 how_many_emplyee = 0 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" Employee.how_many_emplyee += 1 # 請看此行 emp_1 = Employee("Tiny", "Murky", 10000) print(emp_1.how_many_emplyee) # 輸出:1 print(Employee.how_many_emplyee) # 輸出:1 emp_2 = Employee("Test", "User", 10000) print(emp_1.how_many_emplyee) # 輸出:2 print(emp_2.how_many_emplyee) # 輸出:2 print(Employee.how_many_emplyee) # 輸出:2
3. Classmethods and Staticmethods
Classmethods
如果在Class中的function上方加入 @classmethod 的decoration,可以將function變成class method,會讓function的第一個argument直接輸入Class自己(如:Employee)而不是instance。
pythonclass Employee: raise_amt = 1.05 how_many_emplyee = 0 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" Employee.how_many_emplyee += 1 @classmethod def set_raise_amount(cls, amount): # set_raise_amount是class method cls.raise_amt = amount # 此行等同於Employee.raise_amd = amount
以下程式碼表示,在使用 Employee.set_raise_amount(1.1)之後,Class Employee內部的raise_amt改成1.1,並在所有的instance上同步。
pythonemp_1 = Employee("Tiny", "Murky", 10000) emp_2 = Employee("Test", "User", 10000) print(Employee.raise_amt) # 輸出: 1.05 print(emp_1.raise_amt) # 輸出: 1.05 print(emp_2.raise_amt) # 輸出: 1.05 Employee.set_raise_amount(1.1) print(Employee.raise_amt) # 輸出: 1.1 print(emp_1.raise_amt) # 輸出: 1.1 print(emp_2.raise_amt) # 輸出: 1.1
以下是可以但是不太好的寫法,直接在instance上呼叫classmethod也可以直接改到Class的值並影響到所有instance,要注意此方法並無會為instance增加新的raise_amt,而是直接更改Class。
pythonemp_1.set_raise_amount(2.2) print(Employee.raise_amt) # 輸出: 2.2 print(emp_1.raise_amt) # 輸出: 2.2 print(emp_1.__dict__) # 可以看出emp_1.set_raise_amount並不會影響emp_1 # 輸出: {'first': 'Tiny', 'last': 'Murky', # 'pay': 10000, 'email': 'TinyMurky@company.com'} print(emp_2.raise_amt) # 輸出: 2.2
利用Classmethod當作替代的__init__()
classmethod有另一個用法是當作__init__來用,例如今天員工的資料都是用”-”來區分,可以建立一個classmethod Employee.from_string(),這個函式吃Cls和string兩個argument,並會回傳一個cls(first, last, pay),又因為cls就是Class本身,所以效果等同於回傳一個Employee(first, last, pay),直接instance一個物件。
pythonclass Employee: raise_amt = 1.05 how_many_emplyee = 0 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" Employee.how_many_emplyee += 1 @classmethod def from_string(cls, emp_str): first, last, pay = emp_str.split("-") return cls(first, last, pay) # cls就是Employee,所以等同Employee(first,last,pay) emp_str_1 = "Tiny-Murky-10000" emp_1 = Employee.from_string(emp_str_1) print(emp_1.email) # 輸出 TinyMurky@company.com
Staticmethod
Static method是Class中的一種function,他不會接收instance或是class當作argument,所以如果一個function中沒有使用到self或是class,可以盡可能的把他寫成static method
使用static method的時候要在function上面加上@staticmethod 的decoration。
以下的isWeekend輸入星期數,回傳是不是週末,不需要使用任何8faa的self和cls(代表function不需要用到instance或class本身),可以從class中直接叫出來使用,也可以從instance中叫出來。
pythonclass Employee: raise_amt = 1.05 how_many_emplyee = 0 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" Employee.how_many_emplyee += 1 @staticmethod def isWeekend(day): # 未使用self和cls的function if day == 5 or day == 6: return True return False print(Employee.isWeekend(6)) # 輸出 True print(Employee.isWeekend(4)) # 輸出 False
當然也可以直接從instance叫出來用
pythonemp_1 = Employee("Tiny", "Murky", 10000) print(emp_1.isWeekend(6)) # 輸出 True
4. Inheritance — Creating Subclasses
Inheritance(繼承)可以從一個parent class身上得到它所有的attribute和method,並另外加自己的attribute和method變成一個新的class,這樣子在管理上比較方便,不用重複寫很多一樣的code。
以下為本次使用的parent class
pythonclass Employee: raise_amt = 1.05 how_many_emplyee = 0 def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" Employee.how_many_emplyee += 1 def fullname(self): return f"{self.first} {self.last}" def apply_raise(self): self.pay = int(self.pay * self.raise_amt)
接著建立一個class,在class的小括號裏面放上想要從哪個parent繼承。可以從下面的程式碼看到,Developer裏面沒有任何自己的參數,但是卻可以直接instance,這是因為argument會依照繼承鍊(稱為:method resolution order)向parent class 提供值,最後成功instance
pythonclass Developer(Employee): pass dev1 = Developer("Tiny", "Murky", 10000) print(dev1.email) # 輸出 TinyMurky@company.com
method resolution order可以從help(ClassName)中看到,如果要看到Develope的order,可以使用print(help(Developer))
以下為Method resolution order:
- Developer
- Employee
- builtins.object
如果找不到特定的function就會沿著Method resolution order尋找是不是有同名的function,像是在instance的時候,Developer裏面沒有__init__(),因此使用parent class中的function。也就是說在child class當中如果相同名稱的function就會覆寫過parent class。
pythonprint(help(Developer)) """ class Developer(Employee) | Developer(first, last, pay) | | Method resolution order: | Developer | Employee | builtins.object | | Methods inherited from Employee: | | __init__(self, first, last, pay) | Initialize self. See help(type(self)) for accurate signature. | | apply_raise(self) | | fullname(self) | | ---------------------------------------------------------------------- | Data descriptors inherited from Employee: | | __dict__ | dictionary for instance variables (if defined) | | __weakref__ | list of weak references to the object (if defined) | | ----------------------------------------------------------------------
如果子class想要有自己的__init__,可以使用下面的寫法,在Developer Class的__init__當中,我們先放入Parent Class 需要使用的 first, last, pay,接著放入Developer自己的argument “program_language”代表程序員會使用哪種語言。
接著把 first, last, pay放到Parent Class的__init__來完成instance,可以使用super()代表Employee class,super的小括號內可以是空白,或是填上(Developer, self),兩種方法都可以,但要注意Developer是子class名稱不是parent class。也可以使用parent class . init,但是記得在__init__中放入self。
接著把parent class所需以外的argument 放入attribute當中,就可以在parent class既有的attribute外克制化自己的初始值。
從下面的程式碼可以看見,屬於Employee的email和屬於Developer的program_language都已經成功建立。
pythonclass Developer(Employee): def __init__(self, first, last, pay, program_language): super(Developer, self).__init__(first, last, pay) # super().__init__(first, last, pay) 也可以 # Employee.__init__(self, first, last, pay) 也可以 self.program_language = program_language dev1 = Developer("Tiny", "Murky", 10000, "python") print(dev1.email) # 輸出 TinyMurky@company.com print(dev1.program_language) # 輸出 python
除了__init__之外,我們也可以創造Developer自己的method或是覆寫Employer相同的method。
從下方的fullname可以看到覆寫Employee已經有的method,先用super().fullname來呼叫Employee已經寫好的method,回傳全名。在他後面加上使用的程式語言self.program_language後回傳,變成自己的fullname method。
也可以如一般的class一樣自己寫method,像是say_hi是Developer獨有的method,parent class不會有此method。
pythonclass Developer(Employee): def __init__(self, first, last, pay, program_language): super(Developer, self).__init__(first, last, pay) # super().__init__(first, last, pay) 也可以 # Employee.__init__(self, first, last, pay) 也可以 self.program_language = program_language def fullname(self): # 覆寫method return f"{super().fullname()} used {self.program_language}" def say_hi(self): # 自創method return f"{self.last} said hi" dev1 = Developer("Tiny", "Murky", 10000, "python") print(dev1.fullname()) # Tiny Murky used python print(dev1.say_hi()) # Murky said hi print(dev1.__dict__) # 看一下Developer有哪些參數 # {'first': 'Tiny', 'last': 'Murky', 'pay': 10000, # 'email': 'TinyMurky@company.com', 'program_language # ': 'python'}
isinstance() and issubclass()
使用isinstance()可以知道一個物件是不是某個class的instance,下方的dev1是從Developer instance,Dveloper繼承Employee,所以dev1是Dveloper和Employee的 instance,但卻不是同樣繼承Employee的Manager的instance。
使用issubclass()可以知道一個class是否繼承自另一個class,如Developer和Manager都是Employee的subclass,但卻互不為對方的subclass。
pythonclass Manager(Employee): pass dev1 = Developer("Tiny", "Murky", 10000, "python") print( isinstance(dev1, Developer), # True isinstance(dev1, Employee), # True isinstance(dev1, Manager), # False ) print( issubclass(Developer, Employee), # True issubclass(Developer, Manager), # False issubclass(Manager, Manager), # True )
5. Special (Magic/Dunder) Methods 其他雙底線methods
python class 除了__init__()以外還有許多有雙底線的methods,可以參考此Python doc。
這些雙底線的功能可以對應到特定的python預設的 function如len(), str(), repr(), del()。雙底線methods可以讓我們自訂一個class的instance被當作預設function的參數時應該會表現的動作。
repr()與__str__()
python當中設有repr()與str()兩個function,repr()是印出資訊給Developer看,str()則是印出資訊給使用者看。
repl()內的資訊需要寫成像是python程式碼在instance一個class時會長的樣子,這個值如果直接傳給eval()就可以當作程式碼執行。str()則是直接回傳想要給使用者什麼樣的資訊,重點為可讀性。
而__repr__()與__str__()用來應對當instance被當作repr()與str()的參數時,自訂化回傳的資訊。
以下程式碼可以看到如果只設定一個__repr__(),使用print()或是repr()都可以回傳__repr__()的內容。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def __repr__(self): # for developer return f"Employee('{self.first}', '{self.last}', {self.pay})" emp_1 = Employee("Tiny", "Murky", 10000) print(emp_1) # 輸出: Employee('Tiny', 'Murky', 10000) print(repr(emp_1)) # 輸出: Employee('Tiny', 'Murky', 10000) print(emp_1.__repr__()) # 輸出: Employee('Tiny', 'Murky', 10000)
但如果增加__str__(),print()則會回傳__str__()的設定值,str()也會回傳相同值。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def __repr__(self): # for developer return f"Employee('{self.first}', '{self.last}', {self.pay})" def __str__(self): # for end user return f"{self.last}'s email is: '{self.email}'" emp_1 = Employee("Tiny", "Murky", 10000) print(emp_1) #輸出: Murky's email is: 'TinyMurky@company.com' print(repr(emp_1)) # 輸出: Employee('Tiny', 'Murky', 10000) print(emp_1.__repr__())# 輸出: Employee('Tiny', 'Murky', 10000) print(str(emp_1)) #輸出: Murky's email is: 'TinyMurky@company.com' print(emp_1.__str__()) #輸出: Murky's email is: 'TinyMurky@company.com'
add()
Python在使用 + 號時,其實是呼叫物件的__add__() method。可以看到以下程式碼當中數字相加等同於int.add(),文字相加等同於str.add()
pythonprint(1 + 2) # 輸出:3 print(int.__add__(1, 2)) # 輸出:3 print("Tiny " + "Murky") # 輸出:Tiny Murky print(str.__add__("Tiny ", "Murky"))# 輸出:Tiny Murky
因此我們也可以增加__add__()讓Class可以使用 + 號。如下方的程式碼中__add__()將self和另一個Employee的pay加在一起,就可以直接回傳兩人的薪水相加。還可以使用isinstance來檢查+ 號後的另一個是否為Employee type。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def __add__(self, other_guy): # add salary together if isinstance(other_guy, Employee): return self.pay + other_guy.pay return NotImplemented # 如果不是Employee則會出現error emp_1 = Employee("Tiny", "Murky", 10000) emp_2 = Employee("Test", "User", 20000) print(emp_1 + emp_2) #輸出 30000
len()
Python在使用len() function的時候會呼叫物件的__len__(),從以下程式碼可以觀察len(“test”)與”test”.len()都會回傳”test”的長度4。
pythonprint(len("test")) # 輸出:4 print("test".__len__()) # 輸出:4
可以像下方程式碼利用__len__()回傳Employee.fullname()的長度。
pythonclass Employee: def __init__(self, first, last, pay): self.first = first self.last = last self.pay = pay self.email = f"{self.first}{self.last}@company.com" def fullname(self): return f"{self.first} {self.last}" def __len__(self): return len(self.fullname()) emp_1 = Employee("Tiny", "Murky", 10000) print("The length of 'Tiny Murky' is: ", len(emp_1)) #輸出: The length of 'Tiny Murky' is: 10
6. @property, getter, setter and deleter
假設我們有下列的程式碼,如果要取用emp_1的email就要使用emp_1.email這個寫法。但是如果今天想要保護email attribute不希望直接讓別人取用,就需要寫一個get_email(self) 讓別人用method的方法得到email的值。但是可能別的程式碼裏面都已經寫好使用emp_1.email寫法,如果要全部都改成get_email會是一件大工程。我們可以使用@property的寫法來避免大量更改。
pythonclass Employee: def __init__(self, first, last): self.first = first self.last = last self.email = f"{self.first}.{self.last}@company.com" emp_1 = Employee("Tiny", "Smith") @ property 提供 getter, setter 和deleter 3個功能。
getter
getter的功能是當取用class的一個method時可以不用打小括號,如下方的程式碼所顯示,先把__init__的self.email改成self._email做保護,再另寫一個method叫作email(self)回傳self._email,在method上方加上@ property,就可以直接使用emp_1.email呼叫email(self),不需要加小括號。
pythonclass Employee: def __init__(self, first, last): self.first = first self.last = last self._email = f"{self.first}.{self.last}@company.com" @property def email(self): return self._email emp_1 = Employee("Tiny", "Smith") print(emp_1.email) # 輸出: Tiny.Smith@company.com
setter
使用getter之後就不能直接將email用等號的方法改變內容(ex emp_1.email = other.email@company.com),因此我們還需要一個setter。
setter讓我們可以用 “= 號” 來改變emp_1.email的值,使用時需要些建立getter,然後再寫一個與getter同樣名稱的method,method上面需要加上@ getter_name.setter 的字段,才能啟用setter的設定。在setter中寫上要如何處理 = 號 後面的值,並需要有一個argument將其接住。
完成後變可以使用emp_1.email = “other.email@company.com”來改變self._email的值,並使用getter回傳。
pythonclass Employee: def __init__(self, first, last): self.first = first self.last = last self._email = f"{self.first}.{self.last}@company.com" @property def email(self): return self._email @email.setter def email(self, new_email): self._email = new_email emp_1 = Employee("Tiny", "Smith") print(emp_1.email) # 輸出: Tiny.Smith@company.com emp_1.email = "other.email@company.com" print(emp_1.email) # 輸出: other.email@company.com
deleter
若想要刪除emp_1.email的值可以設立deleter,並使用del emp_1.email來刪除(注意不是del(),而是單純的del,沒有小括號)
設立方法需要先建立與getter同名的method,上面加上@getter_name.deleter字段,並在method內部設計如何將self._email清空。
使用時呼叫 del emp_1.email,就可以將self._email清空成None。
pythonclass Employee: def __init__(self, first, last): self.first = first self.last = last self._email = f"{self.first}.{self.last}@company.com" @property def email(self): return self._email @email.setter def email(self, new_email): self._email = new_email @email.deleter def email(self): print("Email Deleted!") self._email = None print(emp_1.email) # 輸出: Tiny.Smith@company.com del emp_1.email # 輸出: Email Deleted! print(emp_1.email) # 輸出: None
結語
這篇文章撰寫的時間比預計的高出很多,希望有幫助到您,謝謝您的閱讀!