黑马程序员技术交流社区

标题: Python基础day09面向对象2(下) [打印本页]

作者: 小江哥    时间: 2018-11-10 15:38
标题: Python基础day09面向对象2(下)
重写父类方法
[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引用的可能是类属性、也有可能是实例属性(这个需要具体分析),不过在存在相同名称的类属性和实例属性的情况下,实例属性优先级更高。静态方法中不需要额外定义参数,因此在静态方法中引用类属性的话,必须通过类对象来引用








欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2