Part6:面向对象
一个典型的类的定义:
class Student:
def __init__(self,name,score): #构造方法第一个参数必须为self
self.name = name #实例属性
self.score = score
def say_score(self): #实例方法
print(self.name,'的分数是:',self.score)
s1 = Student('张三',80) #s1是实例对象,自动调用__init__()方法
s1.say_score()
《__init__构造方法和__new__方法》
一个Python对象包含如下部分:
创建对象,我们需要定义构造函数__init__().构造函数用于执行“实例对象的初始化工作”,即对象创建后,初始化当前对象的相关属性,无返回值。
__new__()方法:用于创建对象,但我们一般无需重新定义该方法。
注: 1. Python 中的self 相当于C++中的self 指针,JAVA 和C#中的this 关键字。Python 中,self 必须为构造函数的第一个参数,名字可以任意修改。但一般遵守惯例,都叫做self。
《实例属性和实例方法》
实例属性
实例属性是从属于实例对象的属性,也称为“实例变量”。他的使用有如下几个要点:
1. 实例属性一般在__init__()方法中通过如下代码定义:
self.实例属性名= 初始值
2. 在本类的其他实例方法中,也是通过self 进行访问:
self.实例属性名
3. 创建实例对象后,通过实例对象访问:
obj01 = 类名() #创建对象,调用__init__()初始化属性
obj01.实例属性名= 值 #可以给已有属性赋值,也可以新加属性
实例方法
要点:
1.定义实例方法时,第一个参数必须为self。和前面一样,self指当前的实例对象。
2.调用实例方法时,不需要也不能给self传参。self由解释器自动传参
函数和方法的区别:
1. 都是用来完成一个功能的语句块,本质一样。
2. 方法调用时,通过对象来调用。方法从属于特定实例对象,普通函数没有这个特点。
3. 直观上看,方法定义时需要传递self,函数不需要。
《类对象、类属性、类方法、静态方法》
类对象:
我们在前面讲的类定义格式中,“class 类名:”。实际上,当解释器执行class 语句时,就会创建一个类对象。
类属性:
类属性从属于类对象,可以被所有实例对象共享
类方法:
类方法是从属于“类对象”的方法。类方法通过@classmethod来定义,
!:类方法中访问实例属性和实例方法会导致错误
静态方法:
与“类对象”无关的方法,称为“静态方法”。
“静态方法”和在模块中定义普通函数没有区别,只不过“静态方法”放到了“类的名字空间里面”,需要通过“类调用”
!:静态方法中访问实例属性和实例方法会导致错误
class Student:
company = "SXT" #类属性
@staticmethod
def add(a,b):
print("{0}+{1}={2}".format(a,b,(a+b)))
return a+b
Student.add(20,10)
《__del__方法(析构函数)和垃圾回收机制》
__del__方法称为“析构方法”,用于实现对象被销毁时所需的操作。比如:释放对象占用的资源,例如:打开的文件资源、网络连接等。
Python 实现自动的垃圾回收,当对象没有被引用时(引用计数为0),由垃圾回收器调用__del__方法。
我们也可以通过del 语句删除对象,从而保证调用__del__方法。
系统会自动提供__del__方法,一般不需要自定义析构方法。
《__call__方法和可调用对象》
定义了__call__方法的对象,称为“可调用对象”,即该对象可以像函数一样被调用。
class SalaryAccount:
def __call__(self, salary):
yearSalary = salary*12
daySalary = salary//30
hourSalary = daySalary // 8
return dict(monthSalary=salary,yearSalary=yearSalary,daySalary=daySalary,hourSalary=hourSalary)
s = SalaryAccount()
print(s(5000)) #可以像调用函数一样调用对象的__call__方法
{'hourSalary': 20, 'daySalary': 166, 'monthSalary': 5000, 'yearSalary': 60000}
《方法没有重载》
如果我们在类体中定义了多个重名的方法,只有最后一个方法有效
《方法的动态性》
,我们可以动态的为类添加新的方法,或者动态的修改类的已有的方法。
class Person:
def __init__(self,name):
self.name = name
def work(self):
print("work!")
def play(s):
print("{0}玩游戏".format(s.name))
def work2(s):
print("work2!")
Person.play = play
Person.work = work2
p = Person("zhangsan")
p.play()
p.work()
《私有属性和私有方法(实现封装)》
1. 通常我们约定,两个下划线开头的属性是私有的(private)。其他为公共的(public)。
2. 类内部可以访问私有属性(方法)
3. 类外部不能直接访问私有属性(方法)
4. 类外部可以通过“_类名__私有属性(方法)名”访问私有属性(方法)
#测试私有属性,私有方法
class Employee:
__company = "wl"
def __init__(self,name,age):
self.name = name
self.__age = age
def say_company(self):
print("我的公司是:",Employee.__company) #类内部可以直接访问私有属性
print(self.name,"的年龄是:",self.__age)
self.__work() #??????
def __work(self):
print("work!")
p1 = Employee("kiki",32)
print(p1.name)
print(dir(p1))
p1.say_company()
print(p1.__Employee_age) #通过这种方式可以访问私有属性
《@property装饰器》
@property 可以将一个方法的调用方式变成“属性调用”。
#测试@property
class Employee:
def __init__(self, name, salary):
self.name = name
self.__salary = salary
@property # 相当于salary 属性的getter 方法
def salary(self):
print("月薪为{0},年薪为{1}".format(self.__salary,(12*self.__salary)))
return self.__salary
@salary.setter
def salary(self, salary): # 相当于salary 属性的setter 方法
if (0 < salary < 1000000):
self.__salary = salary
else:
print("薪水录入错误!只能在0-1000000 之间")
emp1 = Employee("kiki",100)
print(emp1.salary)
emp1.salary = -10