python面向对象编程-多态用法详解

一、面向对象三大特性

封装

  • 将属性和⽅法书写到类的⾥⾯的操作即为封装
  • 封装可以为属性和⽅法添加私有权限

继承

  • ⼦类默认继承⽗类的所有属性和⽅法
  • ⼦类可以重写⽗类属性和⽅法

多态

  • 传⼊不同的对象,产⽣不同的结果

二、多态

2.1 了解多态

多态指的是⼀类事物有多种形态,(⼀个抽象类有多个⼦类,因⽽多态的概念依赖于继承)。
定义:多态是⼀种使⽤对象的⽅式,⼦类重写⽗类⽅法,调⽤不同⼦类对象的相同⽗类⽅法,可以产⽣不同的执⾏结果
好处:调⽤灵活,有了多态,更容易编写出通⽤的代码,做出通⽤的编程,以适应需求的不断变化!
实现步骤:
  • 定义⽗类,并提供公共⽅法
  • 定义⼦类,并重写⽗类⽅法
  • 传递⼦类对象给调⽤者,可以看到不同⼦类执⾏效果不同

2.2 体验多态

示例代码:

class Dog(object):
    def work(self):  # ⽗类提供统⼀的⽅法,哪怕是空⽅法
        print('指哪打哪...')


class ArmyDog(Dog):  # 继承Dog类
    def work(self):  # ⼦类重写⽗类同名⽅法
        print('追击敌⼈...')


class DrugDog(Dog):
    def work(self):
        print('追查毒品...')


class Person(object):
    def work_with_dog(self, dog):  # 传⼊不同的对象,执⾏不同的代码,即不同的work函数
        dog.work()


ad = ArmyDog()
dd = DrugDog()
daqiu = Person()
daqiu.work_with_dog(ad)
daqiu.work_with_dog(dd)

运行结果:

2.3 鸭子类型和多态

示例代码:

"""
鸭子类型:在多个类中实现同一个方法,那么这些类就可以看出同一个类型
"""


class Cat:
    def say(self):
        print("我是猫....")


class Dog:
    def say(self):
        print("我是狗....")


class Duck:
    def say(self):
        print("我是鸭子....")


# 鸭子类型
animal = Cat
animal().say()  # 我是猫....

animal = Dog
animal().say()  # 我是狗....

animal_list = [Cat, Dog, Duck]
for animal in animal_list:
    animal().say()


# 我是猫....
# 我是狗....
# 我是鸭子....


class Student:
    def __init__(self, student_list: list):
        self.student_list = student_list

    def __getitem__(self, item):
        return self.student_list[item]


cls_student = Student(['aa', 'bb'])
new_student = ['cc']

student_tuple = ('dd',)
new_student.extend(student_tuple)
print(new_student)  # ['cc', 'dd']

new_student.extend(cls_student)
print(new_student)  # ['cc', 'dd', 'aa', 'bb']

"""
只要多个类实现了同一个方法,那么就可以归为一类
"""


class Animal:
    def run(self):
        print("动物在跑。。。")


class Dog(Animal):
    def run(self):
        print("狗在跑。。。")


class Cat(Animal):
    def run(self):
        print("猫在跑。。。")


dog = Dog()
dog.run()  # 狗在跑。。。

cat = Cat()
cat.run()  # 猫在跑。。。
"""
在多个类中同时继承了一个类并且对父类中的方法进行了重写
在每个类中所调用的同一个方法会返回不同的行为,这种情况就是多种形态,即多态
"""

运行结果:

. 类属性和实例属性

3.1 类属性

3.1.1 设置和访问类属性

类属性就是 类对象 所拥有的属性,它被 该类的所有实例对象 所共有
类属性可以使⽤ 类对象 实例对象 访问。
class Dog(object):
    tooth = 10


wangcai = Dog()
xiaohei = Dog()
print(Dog.tooth)  # 10
print(wangcai.tooth)  # 10
print(xiaohei.tooth)  # 10
类属性的优点:
记录的某项数据 始终保持⼀致时 ,则定义类属性。
实例属性 要求 每个对象 为其 单独开辟⼀份内存空间 来记录数据,⽽ 类属性 为全类所共有,仅占⽤⼀份内存 更加节省内存空间

3.1.2 修改类属性

        类属性只能通过类对象修改,不能通过实例对象修改,如果通过实例对象修改类属性,表示的是创建了⼀个实例属性。
class Dog(object):
    tooth = 10


wangcai = Dog()
xiaohei = Dog()
# 修改类属性
Dog.tooth = 12
print(Dog.tooth)  # 12
print(wangcai.tooth)  # 12
print(xiaohei.tooth)  # 12
# 不能通过对象修改属性,如果这样操作,实则是创建了⼀个实例属性
wangcai.tooth = 20
print(Dog.tooth)  # 12
print(wangcai.tooth)  # 20
print(xiaohei.tooth)  # 12

3.2 实例属性

class Dog(object):
    def __init__(self):
        self.age = 5

    def info_print(self):
        print(self.age)


wangcai = Dog()
print(wangcai.age)  # 5
# print(Dog.age) # 报错:实例属性不能通过类访问
wangcai.info_print()  # 5

3.3 类属性和实例属性

示例代码:

class A:
    # 类属性
    a = 1

    def __init__(self, b, c):
        # 实例属性
        self.b = b
        self.c = c


a = A(2, 3)

print(a.a, a.b, a.c)  # 1 2 3
print(A.a)  # 1

# 类对象无法访问实例属性
# print(A.b, A.c)  #AttributeError: type object 'A' has no attribute 'b'

# 对类属性的值进行修改
A.a = 11
print(A.a)  # 11
print(a.a)  # 11

# 能否使用实例对象修改类属性
a.a = 22
print(A.a)  # 11
print(a.a)  # 22
"""
实例属性无法修改类属性
当前代码是通过实例对象创建了一个新的实例属性
"""

运行结果:

3.4 类中的三种方法

示例代码:

class Student:
    def __init__(self, name, age, sex):
        self.name = name
        self.age = age
        self.sex = sex

    def add_age(self):
        self.age += 1

    @staticmethod
    def parse_string_1(stu_info):
        name, age, sex = tuple(stu_info.split(','))
        return Student(name, age, sex)

    @classmethod
    def parse_string_2(cls, stu_info):
        name, age, sex = tuple(stu_info.split(','))
        return cls(name, age, sex)

    def __str__(self):
        return f"{self.name, self.age, self.sex}"


stu = Student('dgw', 18, '男')
print(stu)  # ('dgw', 18, '男')
stu.add_age()
print(stu)  # ('dgw', 19, '男')

# 类对象可以访问静态方法
stu2 = Student.parse_string_1('dgw2,25,男')
print(stu2)  # ('dgw2', '25', '男')

stu3 = Student.parse_string_2('dgw3,26,男')
print(stu3)  # ('dgw3', '26', '男')

"""
实例方法:方法中的第一个参数是self,实例方法可以访问类中的所有属性和方法
静态方法:无法直接访问类中的属性
类方法:可以被类对象和实例对象调用,类方法可以访问类属性
"""

"""
使用方法判断:
    如果当前的需求不需要创建实例对象并且不需要访问实例属性/类属性的情况下,可以使用静态方法
    如果当前的需求不需要创建实例对象但是需要访问类属性的情况下,可以使用类方法
    如果需要实例对象并且需要访问类中的所有属性和方法,需要使用实例对象
"""

运行结果:

​​​​​​​

. 类方法和静态方法

4.1 类方法

4.1.1 类方法特点

        需要⽤装饰器 @classmethod 来标识其为类⽅法,对于类⽅法, 第⼀个参数必须是类对象 ,⼀般以 cls 作为第⼀个参数。

4.1.2 类方法使用场景

        当⽅法中 需要使⽤类对象 ( 如访问私有类属性等 ) 时,定义类⽅法
        类⽅法⼀般和类属性配合使⽤
class Dog(object):
    __tooth = 10

    @classmethod
    def get_tooth(cls):
        return cls.__tooth


wangcai = Dog()
result = wangcai.get_tooth()
print(result)  # 10

4.2 静态⽅法

4.2.1 静态⽅法特点

        需要通过装饰器 @staticmethod 来进⾏修饰, 静态⽅法既不需要传递类对象也不需要传递实例对象(形参没有self/cls
        静态⽅法 也能够通过 实例对象 类对象 去访问。

4.2.2 静态⽅法使⽤场景

        当⽅法中 既不需要使⽤实例对象 ( 如实例对象,实例属性 ) 也不需要使⽤类对象 ( 如类属性、类⽅法、创建实例等) 时,定义静态⽅法
        取消不需要的参数传递,有利于 减少不必要的内存占⽤和性能消耗
class Dog(object):
    @staticmethod
    def info_print():
        print('这是⼀个狗类,⽤于创建狗实例....')


wangcai = Dog()
# 静态⽅法既可以使⽤对象访问⼜可以使⽤类访问
wangcai.info_print()
Dog.info_print()

运行结果:

更多python静态方法使用详情,详见博文:python中静态方法staticmethod用法详解_IT之一小佬的博客-CSDN博客

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值