在其他語言中,可以定義多個重名的方法,只要保證方法簽名唯一即可。方法簽名包含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 對於類的成員沒有嚴格的存取控制限制,這與其他物件導向語言有區別。關於私有屬性和私有方法,有如下要點:
【注】方法本質上也是屬性!只不過是可以通過()執行而已。所以,此處講的私有屬性和公有屬性,也同時講解了私有方法和公有方法的用法。如下測試中,同時也包含了私有方法和公有方法的例子。
【測試】私有屬性和公有屬性使用測試
#測試私有屬性、私有方法
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 可以將一個方法的呼叫方式變成「屬性呼叫」
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 主要用於幫助我們處理屬性的讀操作、寫操作。對於某一個屬性,我們可以直接通過類名.方法名 = 值,進行賦值操作。
注:再次強調,方法和屬性都遵循上面的規則