Python 自动化(八)面向对象基础和实践

本文介绍了面向对象编程的基础概念,包括面向过程与面向对象的区别,强调了对象的封装、继承和多态性。讲解了类和对象的定义、关系以及如何使用__init__方法进行初始化。实例演示了如何创建类、对象、属性和方法,以及组合和继承的运用。
摘要由CSDN通过智能技术生成

面向对象编程基础

面向过程

基本概念

面向过程是一种以事件为中心的编程思想,编程的时候把解决问题的步骤分析出来,然后用函数把这些步骤实现,在一步一步的具体步骤中再按顺序调用函数。

关注焦点 —— 怎么做?

  1. 把完成某一个需求的 所有步骤 从头到尾 逐步实现
  2. 根据开发需求,将某些 功能独立 的代码 封装 成一个又一个 函数
  3. 最后完成的代码,就是顺序地调用 不同的函数

特点

  1. 注重 步骤与过程,不注重职责分工
  2. 如果需求复杂,代码会变得很复杂
  3. 开发复杂项目,没有固定的套路,开发难度很大!

面向过程 的设计思路是首先分析 饭店经营 的步骤 — 也就是所有事情自己做!!!:

  1. 买材料() — 采购
  2. 切菜切肉() — 做饭
  3. 端菜() — 服务
  4. 收钱() — 收银结账

用函数实现上面一个一个的步骤,然后在主函数里依次调用上面的函数:

if __name__ == '__main__':
    # 函数下都是我们自己的操作
    # 买材料()
    # 讨价还价()
    # 做饭()
    # 研究新菜谱()
    # 端菜()
    # 收钱()
    # 。。。。。。

而且面向过程不能将顺序打乱,否则会出大乱子,而且所有事情都自己做,估计还能活几天~

面向对象基本概念

  • 我们之前学习的编程方式就是 面向过程
  • 面向过程面向对象,是两种不同的 编程方式
  • 对比 面向过程 的特点,可以更好地了解什么是 面向对象

基本概念

在日常生活或编程中,简单的问题可以用面向过程的思路来解决,直接有效,但是当问题的规模变得更大时,用面向过程的思想是远远不够的。所以慢慢就出现了面向对象的编程思想。世界上有很多人和事物,每一个都可以看做一个对象,而每个对象都有自己的属性和行为,对象与对象之间通过方法来交互。面向对象是一种以“对象”为中心的编程思想,把要解决的问题分解成各个对象,建立对象的目的不是为了完成一个步骤,而是为了描叙某个对象在整个解决问题的步骤中的属性和行为。

关注焦点 —— 谁来做?

相比较函数,面向对象更大封装,根据 职责一个对象中 封装 多个方法

  1. 在完成某一个需求前,首先确定 职责 —— 要做的事情(方法)
  2. 根据 职责 确定不同的 对象,在 对象 内部封装不同的 方法(多个)和属性
  3. 最后完成的代码,就是顺序地让 不同的对象 调用 不同的方法

特点

  1. 注重 对象和职责,不同的对象承担不同的职责
  2. 更加适合应对复杂的需求变化,是专门应对复杂项目开发,提供的固定套路
  3. 需要在面向过程基础上,再学习一些面向对象的语法

采购人员收银小姐姐厨师服务生
工资工资工资工资
采购商品()收钱()做饭()接待顾客()
讨价还价()找零()研究新菜谱()

类和对象

类和对象的概念

对象面向对象编程的 两个 核心概念

  • 是对一群具有 相同 特征 或者 行为 的事物的一个统称,是抽象的,不能直接使用

    • 特征 被称为 属性
    • 行为 被称为 方法
  • 就相当于制造飞机时的图纸,是一个 模板,是 负责创建对象的

对象

  • 对象由类创建出来的一个具体存在,可以直接使用

  • 哪一个类 创建出来的 对象,就拥有在 哪一个类 中定义的:

    • 属性
    • 方法
  • 对象 就相当于用 图纸 制造 的飞机

在程序开发中,应该 先有类,再有对象

类和对象的关系

  • 类是模板对象 是根据 这个模板创建出来的,应该 先有类,再有对象

  • 只有一个,而 对象 可以有很多个

    • 不同的对象 之间 属性 可能会各不相同
  • 中定义了什么 属性和方法对象 中就有什么属性和方法,不可能多,也不可能少

类的设计

在使用面向对象开发前,应该首先分析需求,确定一下,程序中需要包含哪些类!

采购人员收银小姐姐厨师服务生
工资工资工资工资
采购商品()收钱()做饭()接待顾客()
讨价还价()找零()研究新菜谱()

在程序开发中,要设计一个类,通常需要满足一下三个要素:

  1. 类名 这类事物的名字,满足大驼峰命名法
  2. 属性 这类事物具有什么样的特征
  3. 方法 这类事物具有什么样的行为

大驼峰命名法

CapWords

  1. 每一个单词的首字母大写
  2. 单词与单词之间没有下划线

类名的确定

名词提炼法 分析 整个业务流程,出现的 名词,通常就是找到的类

属性和方法的确定

  • 对象的特征描述,通常可以定义成 属性
  • 对象具有的行为(动词),通常可以定义成 方法

提示:需求中没有涉及的属性或者方法在设计类时,不需要考虑

类设计:练习 1

方法:类中定义的函数

函数:类外部定义的函数

需求

  • 小明 今年 18 岁身高 1.75,每天早上 完步,会去 东西
  • 小美 今年 17 岁身高 1.65,小美不跑步,小美喜欢 东西

类设计:练习 2

需求

  • 一只 大型 的泰迪熊玩具
  • 一只 黄颜色 的泰迪熊玩具
  • 这只玩具熊会说话 你好我是泰迪~

面向对象实践

面向对象三大特性

  1. 封装 根据 职责属性方法 封装 到一个抽象的
  2. 继承 实现代码的重用,相同的代码不需要重复的编写
  3. 多态 不同的对象调用相同的方法,产生不同的执行结果,增加代码的灵活度

语法结构

定义只包含方法的类

  • 方法 的定义格式和之前学习过的 函数 的定义几乎一样
  • 区别在于第一个参数必须是 self,大家暂时先记住,稍后介绍 self
class 类名:

    def 方法1(self, 参数列表):
        pass
  
    def 方法2(self, 参数列表):
        pass

创建对象

对象引用 = 类名()

案例

需求

  • 一只玩具熊会说话 你好我是泰迪~

分析

  • 定义一个玩具熊 BearToy
  • 定义一个方法 speak
# 创建类
class BearToy:
    """这是一个玩具熊"""
    def speak(self):
        print("你好我是泰迪~")
# 创建对象
bear = BearToy()
# 通过对象的引用调用方法
bear.speak()

对象引用的说明

  • Python 中使用类 创建对象之后bear 变量中 仍然记录的是 对象在内存中的地址
  • 也就是 bear 变量 引用新建的玩具熊对象

图例

 但是小熊还没有颜色和大小:

# 创建对象
bear = BearToy()
bear.color = "yellow"  # 给熊添加颜色属性并赋值
bear.size = "big"      # 给熊添加大小属性并赋值

方法中的 self 参数

给对象增加属性

  • Python 中,要 给对象设置属性,非常的容易,但是不推荐使用

    • 因为:对象属性的封装应该封装在类的内部
  • 只需要在 类的外部的代码 中直接通过 . 设置一个属性即可

注意:这种方式虽然简单,但是不推荐使用!因为没有加限制,任何属性都可。

# 创建对象
bear = BearToy()
bear.color = "yellow"  # 给熊添加颜色属性并赋值
bear.lunzi = 4         # 给熊添加轮胎的属性,不合逻辑

使用 self 在方法内部输出每一只玩具熊的颜色

哪一个对象 调用的方法,方法内的 self 就是 哪一个对象的引用

  • 在类封装的方法内部,self 就表示 当前调用方法的对象自己

  • 调用方法时,程序员不需要传递 self 参数

  • 在方法内部

    • 可以通过 self. 访问对象的属性
    • 也可以通过 self. 调用其他的对象方法
  • 改造代码如下:

class BearToy:
    def speak(self):
        print("你好我是" + self.color + "色的泰迪~")

bear01 = BearToy()
bear01.color = "red"  # 给熊添加颜色属性并赋值
bear01.speak()

bear02 = BearToy()
bear02.color = "yellow"  # 给熊添加颜色属性并赋值
bear02.speak()

图例

总结

  • 类的外部,通过 变量名. 访问对象的 属性和方法
  • 类封装的方法中,通过 self. 访问对象的 属性和方法

__init__ 初始化方法

问题

  • 将案例代码进行调整,先调用方法 再设置属性,观察一下执行效果
bear01 = BearToy()
bear01.speak()
  • 程序执行报错如下:
AttributeError: 'BearToy' object has no attribute 'color'
属性错误:'BearToy' 对象没有 'color' 属性

提示

  • 在日常开发中,不推荐在 类的外部 给对象增加属性

    • 如果在运行时,没有找到属性,程序会报错
  • 对象应该包含有哪些属性,应该 封装在类的内部

  • 当使用 类名() 创建对象时,会 自动 执行以下操作:

    1. 为对象在内存中 分配空间 —— 创建对象
    2. 为对象的属性 设置初始值 —— 初始化方法(init)
  • 这个 初始化方法 就是 __init__ 方法,__init__ 是对象的内置方法

__init__ 方法是 专门 用来定义一个类 具有哪些属性的方法

BearToy 中增加 __init__ 方法,验证该方法在创建对象时会被自动调用

初始化方法的调用时机

class BearToy:  # 定义类BearToy
    def __init__(self):
        print("对象初始化")
if __name__ == "__main__":
    tidy01 = BearToy()
    tidy02 = BearToy()
    tidy03 = BearToy()

在初始化方法内部定义属性

给小熊的颜色在初始化的时候就确定

class BearToy:  # 定义类BearToy
    def __init__(self):
        print("对象初始化")
        self.color = "yellow"
        self.size = "big"
    def speak():
        print("颜色: " + self.color + ",大小:" + self.size)
if __name__ == "__main__":
    tidy01 = BearToy()
    tidy01.speak()
    tidy02 = BearToy()
    tidy02.speak()

发现颜色和大小都一样,能不能各自有各自的颜色和大小,进一步修改:

class BearToy:  # 定义类BearToy
    def __init__(self, color):
        print("对象初始化")
        self.color = color
    def speak():
        print("颜色: " + self.color + ",大小:" + self.size)
if __name__ == "__main__":
    tidy01 = BearToy("red", "big")
    tidy01.speak()
    tidy02 = BearToy("yellow", "small")
    tidy02.speak()

练习 3:编写游戏人物

需求:

  • 创建游戏角色类
  • 游戏人物角色拥有名字、武器等属性
  • 游戏人物具有攻击的方法
  • 武器通过武器类实现
class Role:                         #定义类Role【拥有相同属性和方法的对象的集合】
    def __init__(self, name, weapon):       #__init__() 可以指定每一个对象独有的属性
        self.name = name            #self 为实例本身的名称
        self.weapon = weapon        #self 为实例本身的名称
#类方法,即类中定义的函数,可以由对象去调用
    def attack(self, target):       #self 为实例本身的名称
        print('我是%s, 正在攻击%s' % (self.name, target))

if __name__ == '__main__':
    lb = Role('吕布', '方天画戟')     #根据Role类创建一个具体的对象lb
    print(lb.name, lb.weapon)       #打印对象lb的名字和武器
    lb.attack('张飞')                #让对象调用类方法attack()

组合

什么是组合

  • 类被定义后,目标就是要把它当成一个模块来使用,并把这些对象嵌入到你的代码中去
  • 组合就是让不同的类混合并加入到其它类中来增加功能和代码重用性
  • 可以在一个大点的类中创建其它类的实例,实现一些其它属性和方法来增强原来的类对象

组合应用

  • 两个类明显不同
  • 一个类是另一个类的组件

案例:组合实践

# 创建新的python文件myclass2.py,类的组合应用
class Weapon:
    def __init__(self, wname, strength):  # __init__() 指定每一个对象独有的属性
        self.wname = wname  # self 为实例本身的名称
        self.strength = strength  # self 为实例本身的名称


class Role:
    def __init__(self, name, weapon):
        self.name = name  # self 为实例本身的名称
        self.weapon = weapon  # self 为实例本身的名称


if __name__ == '__main__':
    ji = Weapon('方天画戟', 100)  # 根据武器类Weapon创建一个具体的对象ji
    lb = Role('吕布', ji)  # 将武器对象ji,作为角色的武器属性
    print(ji.wname, ji.strength)  # 打印武器对象ji的名称和攻击力
    print(lb.weapon.wname, lb.weapon.strength)  # 打印角色对象lb,武器的名称和攻击力

图例

继承

继承的概念、语法和特点

  • 继承的概念子类 拥有 父类 的所有 方法属性

继承的语法

class 类名(父类名):
    pass
  • 子类 继承自 父类,可以直接 享受 父类中已经封装好的方法,不需要再次开发
  • 子类 中应该根据 职责,封装 子类特有的 属性和方法

案例:继承实践

【实践1】

# 创建新的python文件myclass3.py,创建子类【自动继承父类的变量和方法】
class Role:
    def __init__(self, name, weapon):
        self.name = name                #self 为实例本身的名称
        self.weapon = weapon            #self 为实例本身的名称   

    def show_me(self):                  #self 为实例本身的名称
        print('我是%s,我用的武器是%s' % (self.name, self.weapon))

class Warrior(Role):        #创建子类, 小括号()中要写父类的类名【继承父类变量和方法】
    pass

class Mage(Role):           #创建子类Mage, 小括号()中要写父类的类名【继承父类变量和方法】
    pass

if __name__ == '__main__':
    lb = Warrior('吕布', '方天画戟')      #根据子类Warrior,创建一个具体的对象lb 
    km = Mage('孔明', '羽扇')
    lb.show_me()                         #调用子类的实例对象lb和km中的方法show_me()
    km.show_me()

【实践2】

# 在myclass3.py中,子类定义自己的方法
class Role:
    def __init__(self, name, weapon):
        self.name = name                #self 为实例本身的名称
        self.weapon = weapon            #self 为实例本身的名称   
    def show_me(self):                  #self 为实例本身的名称
        print('我是%s,我用的武器是%s' % (self.name, self.weapon))

class Warrior(Role):
    def attack(self, target):       #子类Warrior定义自己的方法
        print('与%s近身肉搏' % target)

class Mage(Role):
    def attack(self, target):           #子类Mage定义自己的方法
        print('远程打击%s' % target)

if __name__ == '__main__':
    lb = Warrior('吕布', '方天画戟 ')
    km = Mage('孔明', '羽扇')
    lb.show_me()
    km.show_me()
    lb.attack('张飞')     #子类的对象调用自己的方法attack()
    km.attack('曹操')

【实践3】

通过继承覆盖方法

  • 如果子类中有和父类同名的方法,父类方法将被覆盖
  • 如果需要访问父类的方法,则要调用一个未绑定的父类方法,明确给出子类的实例
# 在myclass3.py中,编写子类通过继承覆盖父类的方法
class Role:
    def __init__(self, name, weapon):
        self.name = name  # self 为实例本身的名称
        self.weapon = weapon  # self 为实例本身的名称

    def show_me(self):  # self 为实例本身的名称
        print('我是%s,我用的武器是%s' % (self.name, self.weapon))


class Warrior(Role):
    # 方法一:子类通过继承覆盖父类方法
    # def __init__(self, name, weapon, ride):
    #     self.name = name
    #     self.weapon = weapon
    #     self.ride = ride

    # 方法二:子类通过继承覆盖父类方法【推荐使用】
    def __init__(self, name, weapon, ride):
        Role.__init__(self, name, weapon)
        self.ride = ride

    def attack(self, target):
        print('与%s近身肉搏' % target)


class Mage(Role):
    def attack(self, target):
        print('远程打击%s' % target)


if __name__ == '__main__':
    lb = Warrior('吕布', '方天画戟', '赤兔马')  # 子类Warrior多了一个变量ride,需要添加'赤兔马'
    km = Mage('孔明', '羽扇')
    lb.show_me()
    km.show_me()
    lb.attack('张飞')
    km.attack('曹操')

多重继承

  • Python 允许多重继承,即:一个类可以是多个父类的子类,子类可以拥有所有父类的属性
  • 在使用方法时,python有自己的查找顺序:自下向上,自左向右
# 创建新的python文件multi_extend.py,初始代码
class A:                #定义类A, 方法为func1()
    def func1(self):
        print('A func')

class B:
    def func2(self):
        print('B func')

class C(A, B):
    def func3(self):
        print('C func')

if __name__ == '__main__':
    c1 = C()
    c1.func1()
    c1.func2()
    c1.func3()

[root@localhost xxx]# python multi_extend.py
A func
B func
C func

# 类A和B中添加函数func4()
class A:
    def func1(self):
        print('A func')
    def func4(self):        #类A中,包含同样的方法名func4()
        print('A func4')

class B:
    def func2(self):
        print('B func')  
    def func4(self):        #类B中,包含同样的方法名func4() 
        print('B func4')

class C(A, B):
    def func3(self):
        print('C func')

if __name__ == '__main__':
    c1 = C()
    c1.func1()
    c1.func2()
    c1.func3()
    c1.func4()

[root@localhost xxx]# python3 multi_extend.py

类的特殊方法

  • 在 Python 中,所有以 “__” 双下划线包起来的方法,都统称为 “Magic Method”
  • 如果对象实现了这些魔法方法中的某一个,那么这个方法就会在特殊的情况下被 Python 所调用
  • __ init __ 方法:实例化类实例时默认会调用的方法
  • __ str __ 方法:打印/显示实例时调用方法,返回字符串
  • __ call __ 方法:用于创建可调用的实例
# 创建新的python文件books.py,魔法方法 __str__,__call__方法的使用
class Book:     #创建类Book, 定义魔法方法,实现对书籍信息的打印
    def __init__(self, title, author):  #定义__init__方法,获取书籍的信息【默认自动调用】
        self.title = title
        self.author = author
    def __str__(self):                  #定义__str__方法, 必须返回一个字符串
        return  '《%s》' % self.title
    def __call__(self):                 #用于创建可调用的实例,直接作为方法调用
        print('《%s》是%s编写的' % (self.title, self.author))

if __name__ == '__main__':
    pybook = Book('Python核心编程', '韦斯利')      # 抽象出对象pybook
    print(pybook)                           # 调用__str__方法,打印书名
    pybook()                                # 调用实例,默认调用__call__方法

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值