方法没有重载
在其他语言中,可以定义多个重名的方法,只要保证方法签名唯一即可。方法签名包含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 对于类的成员没有严格的访问控制限制,这与其他面向对象语言有区别。关于私有属性和私有方法,有如下要点:
- 通常我们约定,两个下划线("__")开头的属性是私有的(private)。其他为公共的(public)。
- 类内部可以访问私有属性(方法)
- 类外部不能直接访问私有属性(方法)
- 类外部可以通过“_类名__私有属性(方法)名”访问私有属性(方法)
【注】方法本质上也是属性!只不过是可以通过()执行而已。所以,此处讲的私有属性和公有属性,也同时讲解了私有方法和公有方法的用法。如下测试中,同时也包含了私有方法和公有方法的例子。
【测试】私有属性和公有属性使用测试
#测试私有属性、私有方法
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 主要用于帮助我们处理属性的读操作、写操作。对于某一个属性,我们可以直接通过类名.方法名 = 值,进行赋值操作。
属性和方法命名总结
- _xxx:保护成员,不能用“from module import * ”导入,只有类对象和子类对象能访问这些成员。
- __xxx__:系统定义的特殊成员
- __xxx: 类中的私有成员,只有类对象自己能访问,子类对象也不能访问。(但,在类外部可以通过“对象名. _类名__xxx”这种特殊方式访问。Python 不存在严格意义的私有成员)
注:再次强调,方法和属性都遵循上面的规则