重写父类方法
[Python] 纯文本查看 复制代码 class Animal:
def eat(self):
print("-----吃-----")
def drink(self):
print("-----喝-----")
class Dog(Animal):
def bark(self):
print("-----汪汪叫------")
class XTQ(Dog):
"""定义了一个哮天犬 类"""
def bark(self):
print("----嗷嗷叫-----")
class Cat(Animal):
def catch(self):
print("----捉老鼠----")
xtq = XTQ()
xtq.eat()
xtq.bark()
重写父类方法与调用父类方法
[Python] 纯文本查看 复制代码 class Animal(object):
def eat(self):
print("-----吃-----")
def drink(self):
print("-----喝-----")
class Dog(Animal):
def bark(self):
print("-----汪汪叫------")
print("-----汪汪叫------")
print("-----汪汪叫------")
print("-----汪汪叫------")
print("-----汪汪叫------")
class XTQ(Dog):
"""定义了一个哮天犬 类"""
def bark(self):
# print("-----汪汪叫------")
# print("-----汪汪叫------")
# print("-----汪汪叫------")
# print("-----汪汪叫------")
# print("-----汪汪叫------")
# Dog.bark(self) # 调用已经被重写的方法1
super(XTQ, self).bark() # 调用已经被重写的方法2
super().bark() # 调用已经被重写的方法3
print("----嗷嗷叫-----")
class Cat(Animal):
def catch(self):
print("----捉老鼠----")
xtq = XTQ()
xtq.eat()
xtq.bark()
私有方法、属性,继承问题
[Python] 纯文本查看 复制代码 class Animal(object):
def __init__(self):
self.num1 = 1
self.__num2 = 2
def __run(self):
print("----跑---")
def eat(self):
print("-----吃-----")
def drink(self):
print("-----喝-----")
def test(self):
print(self.__num2)
self.__run()
class Dog(Animal):
def bark(self):
print("-----汪汪叫------")
# self.__run() # 父类中的私有方法,没有被子类继承
print(self.num1)
# print(self.__num2) # 父类中的私有属性,没有被子类继承
wang_cai = Dog()
wang_cai.bark()
wang_cai.test()
· 父类中的 私有方法、属性,不会被子类继承
· 可以通过调用继承的父类的共有方法,间接的访问父类的私有方法、属性
[size=21.3333px]
多继承
1. 多继承
从图中能够看出,所谓多继承,即子类有多个父类,并且具有它们的特征
Python中多继承的格式如下:
[Python] 纯文本查看 复制代码 # 定义一个父类class A:
def printA(self):
print('----A----')
# 定义一个父类class B:
def printB(self):
print('----B----')
# 定义一个子类,继承自A、Bclass C(A,B):
def printC(self):
print('----C----')
obj_C = C()
obj_C.printA()
obj_C.printB()
运行结果:
----A----
----B----
说明
· python中是可以多继承的
· 父类中的方法、属性,子类会继承
注意点
· 想一想:
如果在上面的多继承例子中,如果父类A和父类B中,有一个同名的方法,那么通过子类去调用的时候,调用哪个?
[size=10.0000pt]·
[Python] 纯文本查看 复制代码 #coding=utf-8class base(object):
def test(self):
print('----base test----')class A(base):
def test(self):
print('----A test----')
# 定义一个父类class B(base):
def test(self):
print('----B test----')
# 定义一个子类,继承自A、Bclass C(A,B):
pass
obj_C = C()
obj_C.test()
print(C.__mro__) #可以查看C类的对象搜索方法时的先后顺序
多态
多态的概念是应用于Java和C#这一类强类型语言中,而Python崇尚“鸭子类型”。
所谓多态:定义时的类型和运行时的类型不一样,此时就成为多态
· Python伪代码实现Java或C#的多态
[Python] 纯文本查看 复制代码 class F1(object):
def show(self):
print 'F1.show'
class S1(F1):
def show(self):
print 'S1.show'
class S2(F1):
def show(self):
print 'S2.show'
# 由于在Java或C#中定义函数参数时,必须指定参数的类型# 为了让Func函数既可以执行S1对象的show方法,又可以执行S2对象的show方法,所以,定义了一个S1和S2类的父类# 而实际传入的参数是:S1对象和S2对象
def Func(F1 obj):
"""Func函数需要接收一个F1类型或者F1子类的类型"""
print obj.show()
s1_obj = S1()
Func(s1_obj) # 在Func函数中传入S1类的对象 s1_obj,执行 S1 的show方法,结果:S1.show
s2_obj = S2()
Func(s2_obj) # 在Func函数中传入Ss类的对象 ss_obj,执行 Ss 的show方法,结果:S2.show
· Python “鸭子类型”
[Python] 纯文本查看 复制代码 class F1(object):
def show(self):
print 'F1.show'
class S1(F1):
def show(self):
print 'S1.show'
class S2(F1):
def show(self):
print 'S2.show'
def Func(obj):
print obj.show()
s1_obj = S1()
Func(s1_obj)
s2_obj = S2()
Func(s2_obj)
类属性、实例属性
在了解了类基本的东西之后,下面看一下python中这几个概念的区别
先来谈一下类属性和实例属性
在前面的例子中我们接触到的就是实例属性(对象属性),顾名思义,类属性就是类对象所拥有的属性,它被所有类对象的实例对象所共有,在内存中只存在一个副本,这个和C++中类的静态成员变量有点类似。对于公有的类属性,在类外可以通过类对象和实例对象访问
类属性
[Python] 纯文本查看 复制代码 class People(object):
name = 'Tom' #公有的类属性
__age = 12 #私有的类属性
p = People()
print(p.name) #正确
print(People.name) #正确
print(p.__age) #错误,不能在类外通过实例对象访问私有的类属性
print(People.__age) #错误,不能在类外通过类对象访问私有的类属性
实例属性(对象属性)
[Python] 纯文本查看 复制代码 class People(object):
address = '山东' #类属性
def __init__(self):
self.name = 'xiaowang' #实例属性
self.age = 20 #实例属性
p = People()
p.age =12 #实例属性
print(p.address) #正确
print(p.name) #正确
print(p.age) #正确
print(People.address) #正确
print(People.name) #错误
print(People.age) #错误
通过实例(对象)去修改类属性
[Python] 纯文本查看 复制代码 class People(object):
country = 'china' #类属性
print(People.country)
p = People()
print(p.country)
p.country = 'japan'
print(p.country) #实例属性会屏蔽掉同名的类属性
print(People.country)del p.country #删除实例属性
print(p.country)
总结· 如果需要在类外修改类属性,必须通过类对象去引用然后进行修改。如果通过实例对象去引用,会产生一个同名的实例属性,这种方式修改的是实例属性,不会影响到类属性,并且之后如果通过实例对象去引用该名称的属性,实例属性会强制屏蔽掉类属性,即引用的是实例属性,除非删除了该实例属性。[size=21.3333px]
静态方法和类方法
1. 类方法
是类对象所拥有的方法,需要用修饰器@classmethod来标识其为类方法,对于类方法,第一个参数必须是类对象,一般以cls作为第一个参数(当然可以用其他名称的变量作为其第一个参数,但是大部分人都习惯以'cls'作为第一个参数的名字,就最好用'cls'了),能够通过实例对象和类对象去访问。
[Python] 纯文本查看 复制代码 class People(object):
country = 'china'
#类方法,用classmethod来进行修饰 @classmethod
def getCountry(cls):
return cls.country
p = People()print p.getCountry() #可以用过实例对象引用print People.getCountry() #可以通过类对象引用
类方法还有一个用途就是可以对类属性进行修改:
[Python] 纯文本查看 复制代码 class People(object):
country = 'china'
#类方法,用classmethod来进行修饰 @classmethod
def getCountry(cls):
return cls.country
@classmethod
def setCountry(cls,country):
cls.country = country
p = People()print p.getCountry() #可以用过实例对象引用print People.getCountry() #可以通过类对象引用
p.setCountry('japan')
print p.getCountry() print People.getCountry()
结果显示在用类方法对类属性修改之后,通过类对象和实例对象访问都发生了改变
2. 静态方法
需要通过修饰器@staticmethod来进行修饰,静态方法不需要多定义参数
[Python] 纯文本查看 复制代码 class People(object):
country = 'china'
@staticmethod
#静态方法
def getCountry():
return People.country
print People.getCountry()
总结
从类方法和实例方法以及静态方法的定义形式就可以看出来,类方法的第一个参数是类对象cls,那么通过cls引用的必定是类对象的属性和方法;而实例方法的第一个参数是实例对象self,那么通过self引用的可能是类属性、也有可能是实例属性(这个需要具体分析),不过在存在相同名称的类属性和实例属性的情况下,实例属性优先级更高。静态方法中不需要额外定义参数,因此在静态方法中引用类属性的话,必须通过类对象来引用
|