面向对象的概念:
- 类(Class): 用来描述具有相同的属性和方法的对象的集合。它定义了该集合中每个对象所共有的属性和方法。对象是类的实例。
- 类变量:类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用。直接通过类名.属性访问的就是类成员。定义成员在方法外面定义。
- 数据成员:类变量或者实例变量用于处理类及其实例对象的相关的数据。
- 方法重写:如果从父类继承的方法不能满足子类的需求,可以对其进行改写,这个过程叫方法的覆盖(override),也称为方法的重写。类似于java的重写
- 实例变量:定义在方法中的变量,只作用于当前实例的类。调用对象.属性访问的就是对象的变量。写在构造方法里面的成员
- 继承:即一个派生类(derived class)继承基类(base class)的字段和方法。继承也允许把一个派生类的对象作为一个基类对象对待。例如,有这样一个设计:一个Dog类型的对象派生自Animal类,这是模拟"是一个(is-a)"关系(例图,Dog是一个Animal)。
- 实例化:创建一个类的实例,类的具体对象。
- 方法:类中定义的函数。
- 对象:通过类定义的数据结构实例。对象包括两个数据成员(类变量和实例变量)和方法。
和其它编程语言相比,Python 在尽可能不增加新的语法和语义的情况下加入了类机制。
Python中的类提供了面向对象编程的所有基本功能:类的继承机制允许多个基类,派生类可以覆盖基类中的任何方法,方法中可以调用基类中的同名方法。
对象可以包含任意数量和类型的数据。
例如:自定义一个类A,包含属性与方法
# 面向对象练习 class A: # 直接定义的类的变量,属于类(通过对象访问的则是对象的变量) name = '' age = 0 # 私有成员(类似于java的private,python中私有成员和私有方法用__开头) __password = '123456' # 无参构造方法 def __init__(self,height): # 类实例才有的属性 self.height = height print("A被创建") # 有参构造方法 def __init__(self,name,age,password,height): print("A的有参数的构造方法被调用") self.name=name self.age=age self.__password=password # 类实例才有的属性 self.height = height # 成员方法访问私有成员变量 def getPassword(self): return self.__password # 重载方法(类似于java的toString()方法) def __str__(self): return "name:"+self.name+" "+"age:"+str(self.age) def getHeight(self): return self.height def test1(): # 调用有参构造方法创建对象 aObj=A("zhangsan",25,'123456',195) # 打印会调用__str__(self)方法,如果没有重写方法打印的是一串地址 print(aObj) # 访问类实例变量 print(str(aObj.name)+" "+str(aObj.age)+" "+str(aObj.getPassword())) # 访问类变量 print(str(A.age)) #访问类实例才有的属性 print(aObj.height) test1()
结果:
A的有参数的构造方法被调用
name:zhangsan age:25
zhangsan 25 123456
0
195
注意:
(1)类的私有属性
__private_attrs:两个下划线开头,声明该属性为私有,不能在类地外部被使用或直接访问。在类内部的方法中使用时 self.__private_attrs。
(2)类的私有方法
__private_method:两个下划线开头,声明该方法为私有方法,只能在类的内部调用 ,不能在类地外部调用。self.__private_methods。
(3)self与this代表对象自己,一般用self,类似于java的this关键字
(4)在类地内部,使用 def 关键字来定义一个方法,与一般函数定义不同,类方法必须包含参数 self, 且为第一个参数,self 代表的是类的实例。
类的专有方法:
- __init__ : 构造函数,在生成对象时调用
- __del__ : 析构函数,释放对象时使用
- __repr__ : 打印,转换
- __setitem__ : 按照索引赋值
- __getitem__: 按照索引获取值
- __len__: 获得长度
- __cmp__: 比较运算
- __call__: 函数调用
- __add__: 加运算
- __sub__: 减运算
- __mul__: 乘运算
- __div__: 除运算
- __mod__: 求余运算
- __pow__: 乘方
-------------------------------类的继承:------------------------------------
- 继承单个类:
再写一个B类继承A类,并重写方法
# B类继承A类 class B(A): # 在原有属性的基础上增加的属性 sex="男" def __init__(self,name,age,sex): self.name=name self.age=age self.sex=sex # 方法重写(覆盖) def getPassword(self): # super调用父类的方法 print(super().getPassword()) return "这里是B类重写过的方法" #重写方法,写完此方法打印对象不会打印地址 def __str__(self): return "name:"+self.name+" "+"age:"+str(self.age)+" sex:"+self.sex def test2(): bObj = B('wangba',99,"女") # 打印B实例 print(bObj) # 打印B类的属性 print("B类的sex属性:"+B.sex) # B实例的属性 print("B实例的sex属性:"+bObj.sex) #调用B的getPassword方法 print(bObj.getPassword()) test2()
结果:
name:wangba age:99 sex:女
B类的sex属性:男
B实例的sex属性:女
123456
这里是B类重写过的方法
- 继承多个类:
例如重写一个C类继承B类与A类:
class C(B,A): # 属性 idCode='xxx' # 构造方法 def __init__(self,name,age,sex,idCode): self.name=name self.age=age self.sex=sex self.idCode=idCode def test3(): cObj = C('ccc',100,"男",'555') # 实际是调用B类的__str__方法 print(cObj) # 打印类属性 print(C.idCode) # 实际是调用B类的getPassword方法 print(cObj.getPassword()) test3()
注意:
上面只能把B类写在A类前面,否则会报错,因为B类是A类的派生类
需要注意圆括号中父类的顺序,若是父类中有相同的方法名,而在子类使用时未指定,python从左至右搜索 即方法在子类中未找到时,从左到右查找父类中是否包含方法。