Python学习.第十二天.面向对象

本文深入探讨Python面向对象编程的核心概念,包括封装、继承、方法重写和多态。讲解了如何通过封装保护数据安全,使用继承提高代码复用性,实现方法的覆盖,以及利用多态增强程序的可扩展性。同时,文章还详细阐述了Python中的特殊方法如__init__、__str__等,并讨论了类的浅拷贝与深拷贝的区别。
摘要由CSDN通过智能技术生成


前言

  • 封装:提高程序的安全性
    将数据(属性)和行为(方法)包装到类对象中。在方法内部对属性进行操作,在类对象的外部调用方法。这样,无需关心方法内部的具体实现细节,从而隔离了复杂度。
    在Python中没有专门的修饰符用于属性的私有,如果该属性不希望在类对象外部被访问,前边使用两个"_”。
  • 继承:提高代码的复用性
  • 多态:提高程序的可扩展性和可维护性

一、封装

封装:提高程序的安全性

class Student:
    def __init__(self, name, age):
        self.name = name
        self.__age = age  # 年龄不希望在类的外部被使用,所以加两个__  伪私有

    def show(self):
        print(self.name, self.__age)


stu = Student('账单', 20)
stu.show()
# 在类的外部使用name和age
print(stu.name)
# print(stu.__age)  # 会报错
print(dir(stu))
print(stu._Student__age)  # 在类的外部可以通过 _Student__age  进行访问

二、继承

  • 如果一个类没有继承任何类,则默认继承object
  • python支持多继承
  • 定义子类时,必须在其构造函数中调用父类的构造函数
# 单继承
class Person(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def info(self):
        print(self.name, self.age)


class Student(Person):
    def __init__(self, name, age, stu_no):
        super().__init__(name, age)  # 调用父类
        self.stu_no = stu_no


class Teacher(Person):
    def __init__(self, name, age, teacherfyear):
        super().__init__(name, age)
        self.teacherfyear = teacherfyear


# 创建学生和老师的类
stu = Student('漳卅','20','1年纪')
teacher = Teacher('李四','12',21)

# 从person继承过来的 info
stu.info()
teacher.info()
# 多继承
class A(object):
    pass


class B(object):
    pass


class C(A, B):  # C继承A和B
    pass

三、方法重写

  • 如果子类对继承自父类的每个属性或方法不满意,可以在子类对其(方法体)进行重写
  • 子类重写后的方法中可以通过super().xxx()调用父类中被重写的方法
  • 又称方法的覆盖
class Person(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def info(self):
        print('姓名:{0},年龄:{1}'.format(self.name, self.age))


class Student(Person):
    def __init__(self, name, age, score):
        super().__init__(name, age)
        self.score = score

    def info(self):  # 重写
        super().info()  # 先执行父类的输出
        print('学号:{0}'.format(self.score))


stu = Student('Jack', 20, '111')
stu.info()

在这里插入图片描述

四、object

  • object类是所有类的父类,因此所有类都有object类的属性和方法
  • 内置函数dir()可以查看指定对象所有属性
  • Object有一个__str__()方法,用于返回一个对于‘对象的描述’,对应于内置函数str()经常用于对Print()方法,帮我们查看对象的信息,所以我们经常会对_str_()进行重写
class Person(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def info(self):
        print('姓名:{0},年龄:{1}'.format(self.name, self.age))

    def __str__(self):
        return '姓名:{0},年龄:{1}'.format(self.name, self.age)  # {0} 索引 代表name


o = object()
p = Person('Jack', 20)
print(dir(o))  # 查看指定对象的所有属性
print(dir(p))
p.info()
print(p)  # 默认调用__str__()这样的方法

在这里插入图片描述

五、多态的实现

  • 简单说,多态就是“具有多种形态”,指的是,即便不知道一个变量所引用的对象到底是什么类型,仍然可以通过这个变量调用方法,在运行过程中根据变量所引用对象的类型,动态决定调用哪个对象中的方法。
class Animal(object):
    def eat(self):
        print('动物吃东西')


class Dog(Animal):
    def eat(self):
        print('狗狗吃肉')


class Cat(Animal):
    def eat(self):
        print('猫咪吃鱼')


class Fish(object):
    def eat(self):
        print('鱼吃馒头')


class Person:  # 
    def eat(self):
        print('人吃五谷杂粮')


def fun(animal):
    animal.eat()  # 传一个对象进来,只要这个对象有eat方法,就可去执行,跟是否继承无关


fun(Dog())
fun(Cat())
fun(Fish())

fun(Person())  # 不存在继承关系,只是因为Person也有eat方法,所以就可以进行调用输出
  • 静态语言和动态语言

  • 静态语言实现多态的三个必要的条件:继承、方法重写、父类引用指向子类对象

  • 动态语言的多态崇尚"鸭子类型"当看到一只鸟走路起来像鸭子、游泳起来像鸭子、那么这只鸟就可以被称为鸭子。

  • 在鸭子类型中,不需要关心对象是什么类型,到底是不是鸭子,只关系对象的行为。

六、特殊方法和特殊属性

特殊属性:dict ():获得类对象或实例对象所绑定的所有属性和方法的字典

特殊方法:

  • len():通过重写__len__()方法,让内置函数len()的参数可以是自定义类型
  • add():通过重写_add__()方法,可使用自定义对象具有“+”功能
  • new():用于创建对象
  • init():对创建的对象进行初始化
class A:
    pass


class B:
    pass


class C(A, B):
    def __init__(self, name, age):
        self.name = name
        self.age = age


class D(A):
    pass


# 创建C类的对象
x = C('java', 22)  # x是C类型的一个实例对象
print(x.__dict__)  # 实例对象的属性字典
print(C.__dict__)
print('-----------------------------')
print(x.__class__)  # 输出了对象所属的类
print(C.__bases__)  # C类的父类类型的元组
print(C.__base__)  # 类的基类
print(C.__mro__)  # 类的层次结构
print(A.__subclasses__())  # 子类的列表
  • new在前去创造对象 init在后为这个对象的实例属性进行赋值,最后将创建的对象放到P中进行存储。
class Person:

    def __new__(cls, *args, **kwargs):  # 底层person传给cls
        print('__new__被调用执行了,cls的id值为{0}'.format(id(cls)))
        obj = super().__new__(cls)  # 赋值给obj需要占用一个新的内存地址
        print('创建的对象的id为:{0}', format(id(obj)))
        return obj

    def __init__(self, name, age):
        print('__init__被调用了,self的id值为:{0}', format(id(self)))  # obj返回self就是obj的内存地址
        self.name = name
        self.age = age


print('object这个类的对象的id为:{0}', format(id(object)))
print('Person这个类的对象的id为:{0}', format(id(Person)))

# 创建Person的实例对象

P = Person('java', 22)
print('P这个类的对象的id为:{0}', format(id(P)))

# new在前去创造对象 init在后为这个对象的实例属性进行赋值,最后将创建的对象放到P中进行存储,注意看id值

在这里插入图片描述

七、类的浅拷贝与深拷贝

  • 变量的赋值操作:只是形成两个变量,实际上还是指向同一个对象
  • 浅拷贝:Python拷贝一般都是浅拷贝,拷贝是,对象包含的子对象内容不拷贝,因此,源对象与拷贝对象会引用同一个对象
  • 深拷贝:使用copy模块的deepcpoy函数,递归拷贝对象中包含的子对象,源对象个拷贝对象所有的子对象也不相同。

总结

端午休息了几天,坚持,不能死在python门口,把类的浅拷贝的深拷贝跳过了!以后再学~,开启模块学习。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值