01_面向对象
- 面向过程 和 面向对象,是两种不同的 编程方式
过程和函数
- 过程 是早期的一个编程概念,过程 类似于函数,只能执行,但是没有返回值
- 函数 不仅能执行,还可以返回结果
面向过程 和 面向对象 基本概念
1)面向过程 —— 怎么做?
- 把完成某一个需求的 所有步骤 从头到尾 逐步实现
- 根据开发需求,将某些 功能独立 的代码 封装 成一个又一个 函数
- 最后完成的代码,就是顺序地调用 不同的函数
特点
- 注重 步骤与过程,不注重职责分工
- 如果需求复杂,代码会变得很复杂
- 开发复杂项目,没有固定的套路,开发难度很大!
2)面向对象 —— 谁来做?
- 在完成某一个需求前,首先确定 职责 —— 要做的事情(方法)
- 根据 职责 确定不同的 对象,在 对象 内部封装不同的 方法(多个)
- 最后完成的代码,就是顺序地让 不同的对象 调用 不同的方法
特点
- 相比较函数,面向对象 是 更大 的 封装,根据 职责 在 一个对象中 封装 多个方法
- 注重 对象和职责,不同的对象承担不同的职责
- 更加适合应对复杂的需求变化,是专门应对复杂项目开发,提供的固定套路
- 需要在面向过程基础上,再学习一些面向对象的语法
02_类和对象
类和对象的概念
类
- 类 是对一群具有 相同 特征 或者 行为 的事物的一个统称,是抽象的,不能直接使用
i. 特征 被称为 属性
ii. 行为 被称为 方法
对象
- 对象 是 由类创建出来的一个具体存在,可以直接使用
- 由 哪一个类 创建出来的 对象,就拥有在 哪一个类 中定义的:
i. 属性
ii. 方法
类和对象的关系
- 类是模板,对象 是根据 类 这个模板创建出来的,应该 先有类,再有对象
- 类 只有一个,而 对象 可以有很多个,不同的对象 之间 属性 可能会各不相同
- 类 中定义了什么 属性和方法,对象 中就有什么属性和方法,不可能多,也不可能少
类的设计
- 在使用面相对象开发前,应该首先分析需求,确定一下,程序中需要包含哪些类!
- 在程序开发中,要设计一个类,通常需要满足一下三个要素:
i. 类名 这类事物的名字,满足大驼峰命名法
ii. 属性 这类事物具有什么样的特征
iii. 方法 这类事物具有什么样的行为
大驼峰命名法
- 每一个单词的首字母大写
- 单词与单词之间没有下划线
类名的确定
名词提炼法 分析 整个业务流程,出现的 名词,通常就是找到的类
属性和方法的确定
- 对 对象的特征描述,通常可以定义成 属性
- 对象具有的行为(动词),通常可以定义成 方法
03_面向对象基础语法
1)dir 内置函数
- 在 Python 中 对象几乎是无所不在的,之前学习的 变量、数据、函数 都是对象
- 在 标识符 / 数据 后输入一个
.
,然后按下Tab
键,iPython 会提示该对象能够调用的 方法列表 - 使用内置函数
dir
传入 标识符 / 数据,可以查看对象内的 所有属性及方法 - 提示:
__方法名__
格式的方法是 Python 提供的 内置方法 / 属性
方法名 | 类型 |
|
---|---|---|
__new__ | 方法 | 创建对象时,会被 自动 调用 |
__init__ | 方法 | 对象被初始化时,会被 自动 调用 |
__del__ | 方法 | 对象被从内存中销毁前,会被 自动 调用 |
__str__ | 方法 | 返回对象的描述信息,print 函数输出使用 |
def test():
"""
这是一个测试函数
"""
print(111)
test()
print(dir(test))
print(test.__doc__)
- 注意函数名后面不要加括号
2)定义简单的类(只包含方法)
定义只包含方法的类
class 类名:
def 方法1(self, 参数列表):
pass
def 方法2(self, 参数列表):
pass
- 方法 的定义格式和之前学习过的函数 几乎一样
- 区别在于第一个参数必须是
self
- 注意:类名 的 命名规则 要符合 大驼峰命名法
创建对象
- 当一个类定义完成之后,要使用这个类来创建对象
对象变量 = 类名()
第一个面向对象程序
需求
- 小猫 爱 吃 鱼,小猫 要 喝 水
分析
- 定义一个猫类
Cat
- 定义两个方法
eat
和drink
- 按照需求 —— 不需要定义属性
class Cat:
"""这是一个猫类"""
def eat(self):
print("小猫爱吃鱼")
def drink(self):
print("小猫在喝水")
# 创建猫对象
tom = Cat()
# 调用类中的方法
tom.drink()
tom.eat()
引用概念的强调
- 在面向对象开发中,引用的概念是同样适用的!
- 在 Python 中使用类 创建对象之后,
tom
变量中 仍然记录的是 对象在内存中的地址 - 也就是
tom
变量 引用 了 新建的猫对象 - 使用
print
输出 对象变量,默认情况下,是能够输出这个变量 引用的对象 是 由哪一个类创建的对象,以及 在内存中的地址(十六进制表示)
案例进阶 —— 使用 Cat
类再创建一个对象
class Cat:
""" 这是猫类"""
def eat(self):
print("小猫爱吃鱼")
def drink(self):
print("小猫在喝水")
# 创建猫对象
Tom = Cat()
# 调用类中的方法
Tom.eat()
Tom.drink()
print(Tom)
# 再创建一个猫对象
lazy_cat = Cat()
lazy_cat.eat()
lazy_cat.drink()
print(lazy_cat)
lazy_cat2 = lazy_cat
print(lazy_cat2)
- 注意:
tom
和lazy_cat
不是同一个对象,lazy_cat
和lazy_cat2
是同一个对象
3)方法中的 self
参数
案例改造 —— 给对象增加属性
- 在 Python 中,要 给对象设置属性,非常的容易,但是不推荐使用,因为:对象属性的封装应该封装在类的内部
- 只需要在 类的外部的代码 中直接通过 . 设置一个属性即可
tom.name = "Tom"
...
lazy_cat.name = "大懒猫"
使用 self 在方法内部输出每一只猫的名字
- 由 哪一个对象 调用的方法,方法内的
self
就是 哪一个对象的引用 - 在类封装的方法内部,
self
就表示 当前调用方法的对象自己 - 调用方法时,程序员不需要传递
self
参数 - 在方法内部
i. 可以通过self.
访问对象的属性
ii. 也可以通过self.
调用其他的对象方法
class Cat:
def eat(self):
print("%s 爱吃鱼" % self.name)
tom = Cat()
tom.name = "Tom"
tom.eat()
lazy_cat = Cat()
lazy_cat.name = "大懒猫"
lazy_cat.eat()
- 在 类的外部,通过 变量名. 访问对象的 属性和方法
- 在 类封装的方法中,通过
self.
访问对象的 属性和方法
4)初始化方法
之前代码存在的问题 —— 在类的外部给对象增加属性
class Cat:
def eat(self):
print("%s 爱吃鱼" % self.name)
tom = Cat()
tom.drink()
tom.eat()
tom.name = "Tom"
print(tom)
- 注意:先调用方法 再设置属性,程序会报错
- 对象应该包含的属性,应该 封装在类的内部
初始化方法
-
当使用 类名() 创建对象时,会 自动 执行以下操作:
i. 为对象在内存中 分配空间 —— 创建对象
ii. 为对象的属性 设置初始值 —— 初始化方法(init
) -
这个 初始化方法 就是
__init__
方法,__init__
是对象的内置方法 -
__init__
方法是 专门 用来定义一个类 具有哪些属性的方法!
class Cat:
def __init__(self):
print("这是一个初始化方法")
# 使用类目()创建对象的时候,会自动调用初始化方法__init__
Tom = Cat()
在初始化方法内部定义属性
- 在
__init__
方法内部使用self.
属性名 = 属性的初始值 就可以 定义属性 - 定义属性之后,再使用
Cat
类创建的对象,都会拥有该属性
class Cat:
def __init__(self):
print("这是一个初始化方法")
# self.属性名 = 属性的初始值
self.name = "Tom"
def eat(self):
print("%s 爱吃鱼" % self.name)
Tom = cat()
print(Tom.name)
Tom.eat()
改造初始化方法 —— 初始化的同时设置初始值
在开发中,如果希望在 创建对象的同时,就设置对象的属性,可以对 __init__
方法进行 改造
- 把希望设置的属性值,定义成
__init__
方法的参数 - 在方法内部使用
self.
属性 = 形参 接收外部传递的参数 - 在创建对象时,使用 类名(属性1, 属性2…) 调用
class Cat:
def __init__(self, name):
print("这是一个初始化方法")
# self.属性名 = 属性的初始值
self.name = name
def eat(self):
print("%s 爱吃鱼" % self.name)
Tom = cat("Tom")
print(Tom.name)
Tom.eat()
lazy_cat = cat("懒猫")
print(lazy_cat.name)
lazy_cat.eat()
5)内置方法和属性
方法名 | 类型 |
|
---|---|---|
__del__ | 方法 | 对象被从内存中销毁前,会被 自动 调用 |
__str__ | 方法 | 返回对象的描述信息,print 函数输出使用 |
__del__
方法
在 Python 中
- 当使用
类名()
创建对象时,为对象 分配完空间后,自动 调用__init__
方法 - 当一个 对象被从内存中销毁 前,会 自动 调用
__del__
方法
应用场景
__init__
改造初始化方法,可以让创建对象更加灵活__del__
如果希望在对象被销毁前,再做一些事情,可以考虑一下
生命周期
- 一个对象从调用 类名() 创建,生命周期开始
- 一个对象的
__del__
方法一旦被调用,生命周期结束 - 在对象的生命周期内,可以访问对象属性,或者让对象调用方法
class Cat:
def __init__(self, name):
self.name = name
print("%s 来了" % self.name)
def __del__(self):
print("%s 走了" % self.name)
# tom 是一个全局变量
tom = Cat("Tom")
print(tom.name)
# del 关键字可以删除一个对象
del tom
print("-" * 50)
__str__
方法
- 在 Python 中,使用
print
输出 对象变量,默认情况下,会输出这个变量 引用的对象 是 由哪一个类创建的对象,以及 在内存中的地址(十六进制表示) - 如果在开发中,希望使用
print
输出 对象变量 时,能够打印 自定义的内容,就可以利用__str__
这个内置方法了 - 注意:
__str__
方法必须返回一个字符串
class Cat:
def __init__(self, name):
self.name = name
print("%s 来了" % self.name)
def __del__(self):
print("%s 走了" % self.name)
def __str__(self):
# str 方法必须返回一个字符串
return "我是小猫:%s" % self.name
# tom 是一个全局变量
tom = Cat("Tom")
print(tom)