Python 物件導向

2022-12-30 06:01:02

Python 物件導向

方法沒有過載

在其他語言中,可以定義多個重名的方法,只要保證方法簽名唯一即可。方法簽名包含3個部分:方法名、引數數量、引數型別。

Python 中,方法的的引數沒有宣告型別(呼叫時確定引數的型別),引數的數量也可以由
可變引數控制。因此,Python 中是沒有方法的過載的。定義一個方法即可有多種呼叫方式,相當於實現了其他語言中的方法的過載。

如果我們在類體中定義了多個重名的方法,只有最後一個方法有效。

建議:不要使用重名的方法!Python 中方法沒有過載。

#Python 中沒有方法的過載。定義多個同名方法,只有最後一個有效

class Person:

    def say_hi(self):
        print("hello")

    def say_hi(self,name):
        print("{0},hello".format(name))

p1 = Person()

#p1.say_hi() #不帶參,報錯:TypeError: say_hi() missing 1 required positional argument: 'name' 

p1.say_hi("陳浩")

方法的動態性

Python 是動態語言,我們可以動態的為類新增新的方法,或者動態的修改類的已有的方法。

#測試方法的動態性
class Person:
    def work(self):
        print("努力上班!")

def play_game(self):
    print("玩遊戲")

def work2(self):
    print("好好工作,努力上班!")

Person.play = play_game
Person.work = work2
p = Person()

p.play()    # 玩遊戲
p.work()    # 好好工作,努力上班!

我們可以看到,Person 動態的新增了 play_game 方法,以及用 work2 替換了 work 方法。

私有屬性和私有方法(實現封裝)

Python 對於類的成員沒有嚴格的存取控制限制,這與其他物件導向語言有區別。關於私有屬性和私有方法,有如下要點:

  1. 通常我們約定,兩個下劃線("__")開頭的屬性是私有的(private)。其他為公共的(public)。
  2. 類內部可以存取私有屬性(方法)
  3. 類外部不能直接存取私有屬性(方法)
  4. 類外部可以通過「_類名__私有屬性(方法)名」存取私有屬性(方法)

【注】方法本質上也是屬性!只不過是可以通過()執行而已。所以,此處講的私有屬性和公有屬性,也同時講解了私有方法和公有方法的用法。如下測試中,同時也包含了私有方法和公有方法的例子。

【測試】私有屬性和公有屬性使用測試

#測試私有屬性、私有方法
class Employee: 
    __company = "Danone" #私有類屬性. 通過 dir 可以查到_Employee__company
    
    def __init__(self, name, age):
        self.name = name
        self.__age = age #私有範例屬性

    def say_company(self):
        print("我的公司是:",Employee.__company) #類內部可以直接存取私有屬性
        print(self.name,"的年齡是:",self.__age)
        
    def __work(self): #私有實體方法 通過 dir 可以查到_Employee__work
        return "工作!好好工作,好好賺錢,娶個媳婦!"

p1 = Employee("陳浩",18)

print(p1.name) 
# 陳浩

print(dir(p1)) 
# ['_Employee__age', '_Employee__company', '_Employee__work', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__getstate__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'say_company']

p1.say_company()
# 我的公司是: Danone
# 陳浩 的年齡是: 18

print(p1._Employee__age) #通過這種方式可以直接存取到私有屬性 。通過 dir 可以查到屬性:_Employee__age
# 18

print(p1._Employee__work())
# 工作!好好工作,好好賺錢,娶個媳婦!

#print(p1.__age) #直接存取私有屬性,報錯
#print(p1.__work()) #直接存取私有方法,報錯

@property 裝飾器

@property 可以將一個方法的呼叫方式變成「屬性呼叫」

class Employee:

    def __init__(self, name, salary):
        self.name = name
        self.__salary = salary


    @property               #getter方法 
    def salary(self):
        return self.__salary

    @salary.setter          #setter方法
    def salary(self, salary):
        self.__salary = salary

    @salary.deleter         #deleter方法
    def salary(self):
        self.__salary = 0



emp = Employee("陳浩", 6000)
print("{0}當前的薪資是{1}".format(emp.name,emp.salary))
# 陳浩當前的薪資是6000

print("*"*12, "加薪啦", "*"*12)
emp.salary = 10000
print("{0}當前的薪資是{1}".format(emp.name,emp.salary))
# 陳浩當前的薪資是10000

print("*"*12, "離職了", "*"*12)
del emp.salary
print("{0}當前的薪資是{1}".format(emp.name,emp.salary))
# 陳浩當前的薪資是0

@property 主要用於幫助我們處理屬性的讀操作、寫操作。對於某一個屬性,我們可以直接通過類名.方法名 = 值,進行賦值操作。

屬性和方法命名總結

  1. _xxx:保護成員,不能用「from module import * 」匯入,只有類物件和子類物件能存取這些成員。
  2. __xxx__:系統定義的特殊成員
  3. __xxx: 類中的私有成員,只有類物件自己能存取,子類物件也不能存取。(但,在類外部可以通過「物件名. _類名__xxx」這種特殊方式存取。Python 不存在嚴格意義的私有成員)

注:再次強調,方法和屬性都遵循上面的規則