Python基础day09面向对象2(下)

80 阅读8分钟
重写父类方法


[Python]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
# 定义一个父类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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
#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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
class People(object):
name = 'Tom' #公有的类属性
__age = 12 #私有的类属性
p = People()
print(p.name) #正确
print(People.name) #正确
print(p.__age) #错误,不能在类外通过实例对象访问私有的类属性
print(People.__age) #错误,不能在类外通过类对象访问私有的类属性


实例属性
(对象属性)


[Python]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
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]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
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]
纯文本查看
复制代码
1
2
3
4
5
6
7
8
class People(object):
country = 'china'
#类方法,用classmethod来进行修饰 @classmethod
def getCountry(cls):
return cls.country
p = People()print p.getCountry() #可以用过实例对象引用print People.getCountry() #可以通过类对象引用



类方法还有一个用途就是可以对类属性进行修改:


[Python]
纯文本查看
复制代码
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
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]
纯文本查看
复制代码
1
2
3
4
5
6
7
8
class People(object):
country = 'china'
@staticmethod
#静态方法
def getCountry():
return People.country
print People.getCountry()




总结


从类方法和实例方法以及静态方法的定义形式就可以看出来,类方法的第一个参数是类对象
cls,那么通过cls引用的必定是类对象的属性和方法;而实例方法的第一个参数是实例对象self,那么通过self引用的可能是类属性、也有可能是实例属性(这个需要具体分析),不过在存在相同名称的类属性和实例属性的情况下,实例属性优先级更高。静态方法中不需要额外定义参数,因此在静态方法中引用类属性的话,必须通过类对象来引用

更多免费技术资料可关注:annalin1203