以下面代码为例,分析整个创建过程:
class Student:
company = '尚学堂' #类属性
count = 0 #类属性
def __init__(self,name,score):
self.name = name #实例属性
self.score = score
Student.count = Student.count + 1
def say_score(self): #实例方法
print("我的公司是:", Student.company)
print(self.name,'的分数是:', self.score)
s1 = Student('高琪',80) #s1是
实例对象,自动调用__init__()方法
s1.say_score()
print('一共创建{0}个对象'.format(Student.count))
1、调用类是,先创建一个Student类的类对象,类属性和方法会被加载到类对象中;
2、调用构造器 __ init __()把对象构造起来,对象中包括实例属性和方法
3、两个实例属性传入,方法从类对象中获得
4、如果多一个s2则重复2-3的过程,即多一个 __ init __()把对象构造起来的过程,类对象还是只有一个。
__ del __方法(析构函数)和垃圾回收机制
__ del __方法称为“析构方法”,用于实现对象被销毁时所需的操作。比如:释放对象占用的资源,例如:打开的文件资源、网络连接等。
Python实现自动的垃圾回收,当对象没有被引用时(引用计数为0),由垃圾回收器调用 __ del __方法。
我们也可以通过del语句删除对象,从而保证调用 __ del __方法。
系统会自动提供 __ del __方法,一般不需要自定义析构方法。
__ cal __方法和可调用对象
定义了__ 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))
方法没有重载
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('高琪')
方法的动态性
Python是动态语言,我们可以动态的为类添加新的方法,或者动态的修改类的已有的方法。
#测试方法的动态性
class Person:
def work(self):
print("努力上班!")
def play_game(self):
print("{0}玩游戏".format(self))
def work2(s):
print("好好工作,努力上班!")
Person.play = play_game
Person.work = work2
p = Person()
p.play()
p.work()
执行结果:
<__main__.Person object at 0x0000000002F634A8>玩游戏
好好工作,努力上班!
我们可以看到,Person动态的新增了play_game方法,以及用work2替换了work方法。
私有属性和私有方法(实现封装)
Python对于类的成员没有严格的访问控制限制,这与其他面向对象语言有区别。关于私有属性和私有方法,有如下要点:
1、通常我们约定,两个下划线开头的属性是私有的(private)。其他为公共的(public)。
2、类内部可以访问私有属性(方法)。
3、类外部不能直接访问私有属性(方法)。
4、类外部可以通过“类名_私有属性(方法)名”访问私有属性(方法)。
【注】方法本质上也是属性!只不过是可以通过()执行而已。所以,此处讲的私有属性和共有属性,也同时讲解了私有方法和公有方法的用法。
【测试】私有属性和公有属性使用测试
class Employee:
__company = "百战程序员" #私有类属性, 通过dir可以查到
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): #私有实例方法,通过dir可以查到
print("工作!好好工作,好好挣钱,去个媳妇!")
p1 = Employee("高琪",32)
print(p1.name)
print(dir(p1))
p1.say_company()
print(p1._Employee__age) #通过这种方式可以直接访问到私有属性。通过dir可以查到属性:_Employee__age
执行结果:
高琪
['_Employee__age', '_Employee__company', '_Employee__work', '__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'say_company']
我的公司是: 百战程序员
高琪 的年龄是: 32
工作!好好工作,好好挣钱,去个媳妇!
32
@property装饰器
@property可以将一个方法的调用方式变成“属性调用”。
#简单测试@property
class Employee:
@property
def salary(self):
return 30000;
emp1 = Employee()
print(emp1.salary)
print(type(emp1.salary))
执行结果:
30000
<class 'int'>
#简单测试@property
class Employee:
def __init__(self,name,salary):
self.name = name
self.__salary = salary
@property
def salary(self): #相当于salary属性的getter方法
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~100000之间')
emp1 = Employee('高琪',100)
print(emp1.salary)
print(type(emp1.salary))
emp1.salary = -200
执行结果:
月薪为100,年薪为1200
100
月薪为100,年薪为1200
<class 'int'>
薪水录入错误!只能在0~100000之间