Python基础——面向对象(三)

1.继承

  • 继承是⾯向对象三⼤特性之⼀
  • 通过继承我们可以使⼀个类获取到其他类中的属性和⽅法
  • 在定义类时,可以在类名后⾯的括号中指定当前类的⽗类(超类、基类)
  • 继承提⾼了类的复⽤性。让类与类之间产⽣了关系。有了这个关系,才有了多态的特性
# 继承
class Job():
    # 属性
    name = '工作'
    # 方法
    def content(self):
        print('工作内容...')
    def work_time(self):
        print('工作时间...')
    def salary(self):
        print('工作薪资...')

class Programmers(Job):
    pass
# 实例化两个类
myjob = Job()
programmer = Programmers()
# 子类具有和父类相同的属性和方法
print(myjob.name)       #工作
print(programmer.name)  #工作
myjob.content()         #工作内容...
programmer.content()    #工作内容...

2.重写

  • 如果在⼦类中有和⽗类同名的⽅法,则通过⼦类实例去调⽤⽅法时,会调⽤⼦类的⽅法⽽不是⽗类的⽅法,这个特点我们称之为⽅法的重写(覆盖)
  • 当我们调⽤⼀个对象的⽅法时:
    • 会优先去当前对象中寻找是否具有该⽅法,如果有则直接调⽤
    • 如果没有,则去当前对象的⽗类中寻找,如果⽗类中有则直接调⽤⽗类中的⽅法
    • 如果没有,则去⽗类中的⽗类寻找,以此类推,直到找到object,如果依然没有找到就报错了
# 重写
class Job():
    # 属性
    name = '工作'
    # 方法
    def content(self):
        print('工作内容...')
    def work_time(self):
        print('工作时间...')
    def salary(self):
        print('工作薪资...')

class Worker(Job):
    def work_time(self):
        print('朝九晚五...')

class Programmers(Worker):
    # 子类定义和父类相同的方法,即重写
    def content(self):
        print('敲键盘...')

# 实例化一个类
programmer = Programmers()

# 优先去当前对象中寻找是否具有该⽅法,如果有则直接调⽤
programmer.content()    #敲键盘...
# 如果没有,则去当前对象的⽗类中寻找,如果⽗类中有则直接调⽤⽗类中的⽅法
programmer.work_time()  #朝九晚五...
# 如果没有,则去⽗类中的⽗类寻找
programmer.salary()     #工作薪资...
# 以此类推,直到找到object,如果依然没有找到就报错了
programmer.others()     #AttributeError: 'Programmers' object has no attribute 'others'

3.super()

  • super()可以获取当前类的⽗类
  • 并且通过super()返回对象调⽤⽗类⽅法时,不需要传递self
# super()
class Fruits():
    def __init__(self,name):
        self._name = name
    def taste(self):
        print('太好吃了...')

class Apple(Fruits):
    def __init__(self,name,number):
        # 如果类名更换,下面的语句便会报错,故采取动态的调用方式,即super()
        # Fruits.__init__(self,name)

        # super()可以获取当前类的父类
        super().taste()
        # 用super()做初始化
        super().__init__(name)
        self._number = number

# 实例化一个类
fruit = Apple('apple',5)        #太好吃了...

4.多重继承

  • 在Python中是⽀持多重继承的。也就是我们可以为⼀个类同时制定多个⽗类
  • 可以在类名的()后边添加多个类,来实现多重继承
  • 多重继承,会使⼦类同时拥有多个⽗类,并且会获取到所有⽗类中的⽅法
  • 在开发中没有特殊情况,应该尽量避免使⽤多重继承。因为多重继承会让我们的代码更加复杂
  • 如果多个⽗类中有同名的⽅法,则会先在第⼀个⽗类中寻找,然后找第⼆个,找第三个…前⾯会覆盖后⾯的方法
# 多重继承
class Bear():
    def sleep(self):
        print('熊睡觉了...')
    def run(self):
        print('熊跑了...')

class Cat():
    def sleep(self):
        print('猫睡觉了...')
    def jump(self):
        print('猫跳走了...')

class Panda(Bear,Cat):
    # 在类名的()里面添加多个类,来实现多重继承
    def eat(self):
        print('熊猫吃东西了...')

# 实例化一个类
animal = Panda()
# ⼦类同时拥有多个⽗类时,会获取到所有⽗类中的⽅法,以及自己的方法
animal.run()            #熊跑了...
animal.jump()           #猫跳走了...
animal.eat()            #熊猫吃东西了...
# 如果多个⽗类中有同名的⽅法,则会先在第⼀个⽗类中寻找,然后找第⼆个,找第三个…前⾯会覆盖后⾯的方法
animal.sleep()          #熊睡觉了...

5.多态

  • 多态是⾯向对象的三⼤特性之⼀。从字⾯理解就是多种形态
  • ⼀个对象可以以不同形态去呈现
  • ⾯向对象三⼤特性
    • 封装 确保对象中数据的安全
    • 继承 保证了对象的扩展性
    • 多态 保证了程序的灵活
class Dog():
    def __init__(self,age):
        self._name = '狗'
        self._age = age
    @property
    def name(self):
        return self._name
    @property
    def age(self):
        return self._age

class Cat():
    def __init__(self,age):
        self._name = '猫'
        self._age = age
    @property
    def name(self):
        return self._name
    @property
    def age(self):
        return self._age

# 具有相同属性(name和age)的Dog()类和Cat()类能够在下列函数运行
def info(animal):
    if isinstance(animal,Cat) or isinstance(animal,Dog):
        print(animal.name+'已经'+str(animal.age)+'岁了!!!')
    else:
        print('%s不在规定的动物列表中'%animal)
# 实例化两个类并定义一个字符串
dog = Dog(2)
cat = Cat(3)
mice = '老鼠'
# 对上述对象调用函数
info(dog)           #狗已经2岁了!!!
info(cat)           #猫已经3岁了!!!
info(mice)          #老鼠不在规定的动物列表中

6.属性和方法

  • 属性
    • 类属性,直接在类中定义的属性是类属性
    • 类属性可以通过类或类的实例访问到。但是类属性只能通过类对象来修改,⽆法通过实例对象修改
    • 实例属性 通过实例对象添加的属性属于实例属性
    • 实例属性只能通过实例对象来访问和修改,类对象⽆法访问修改
#属性
class Person():
    # 类属性,直接在类中定义的属性是类属性
    name = 'Nathanil'

# 实例化对象
me = Person()
me.name = 'Tom'
# 类属性可以通过类或类的实例访问到
# 但是类属性只能通过类对象来修改,⽆法通过实例对象修改
print(Person.name,'&',me.name)      #Nathanil & Tom

# 通过实例对象添加的属性属于实例属性
# 实例属性只能通过实例对象来访问和修改,类对象⽆法访问修改
me.age = '6'
# print(Person.age)                 #AttributeError: type object 'Person' has no attribute 'age'
print(me.age)                       #6
  • ⽅法
    • 在类中定义,以self为第⼀个参数的⽅法都是实例⽅法 实例⽅法在调⽤时,Python会将调⽤对象以self传⼊
    • 实例⽅法可以通过类实例和类去调⽤
    • 当通过实例调⽤时,会⾃动将当前调⽤对象作为self传⼊
    • 当通过类调⽤时,不会⾃动传递self,我们必须⼿动传递self
    • 类⽅法 在类的内容以@classmethod 来修饰的⽅法属性类⽅法
    • 类⽅法第⼀个参数是cls 也会⾃动被传递。cls就是当前的类对象
    • 类⽅法和实例⽅法的区别,实例⽅法的第⼀个参数是self,类⽅法的第⼀ 个参数是cls
    • 类⽅法可以通过类去调⽤,也可以通过实例调⽤
# ⽅法
class Animal():
    # 在类中定义,以self为第⼀个参数的⽅法都是实例⽅法 实例⽅法在调⽤时,Python会将调⽤对象以self传⼊
    def sleep(self):
        print('睡觉了')

    # 类⽅法 在类的内容以@classmethod 来修饰的⽅法属性类⽅法
    @classmethod
    # 类⽅法第⼀个参数是cls 也会⾃动被传递。cls就是当前的类对象
    # 类⽅法和实例⽅法的区别,实例⽅法的第⼀个参数是self,类⽅法的第⼀ 个参数是cls
    def run(cls):
        print('跑了')

# 实例化对象
dog = Animal()
# 实例⽅法可以通过类实例和类去调⽤
# 当通过实例调⽤时,会⾃动将当前调⽤对象作为self传⼊
dog.sleep()                 #睡觉了
# 当通过类调⽤时,不会⾃动传递self,我们必须⼿动传递self
# Animal.sleep()            #TypeError: sleep() missing 1 required positional argument: 'self'
Animal.sleep(dog)           #睡觉了

# 类⽅法可以通过类去调⽤,也可以通过实例调⽤
dog.run()                   #跑了
Animal.run()                #跑了
  • 静态⽅法
    • 在类中⽤@staticmethod来修饰的⽅法属于静态⽅法
    • 静态⽅法不需要指定任何的默认参数,静态⽅法可以通过类和实例调⽤
    • 静态⽅法,基本上是⼀个和当前类⽆关的⽅法,它只是⼀个保存到当前类中的函数
    • 静态⽅法⼀般都是些⼯具⽅法,和当前类⽆关
# 静态⽅法
# 静态⽅法⼀般都是些⼯具⽅法,和当前类⽆关
class Plant():
    # 在类中⽤@staticmethod来修饰的⽅法属于静态⽅法
    @staticmethod
    def grow():
        # 静态⽅法,基本上是⼀个和当前类⽆关的⽅法,它只是⼀个保存到当前类中的函数
        print('成长ing...')

tree = Plant()
# 静态⽅法不需要指定任何的默认参数,静态⽅法可以通过类和实例调⽤
tree.grow()         #成长ing...
Plant().grow()      #成长ing...
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值