面向对象(OOP)
面向对象编程——Object Oriented Programing
简写 OOP
面向对象基本概念
- 我们之前学习的编程方式就是面向过程的
- 面向过程和面向对象,是两种不同的编程方式
- 对比面向过程的特点,可以更好地了解什么是面向对象
面向过程和面向对象基本概念
面向过程
- 把完成某一个需求的所有步骤从头到尾逐步实现
- 根据开发需求,将某些功能独立的代码封装成一个又一个函数
- 最后完成的代码,就是顺序地调用不同的函数
特点
- 注重步骤与过程,不注重职责分工
- 如果需求复杂,代码会变得很复杂
- 开发复杂项目,没有固定的套路,开发难度很大
面向对象
相比较函数,面向对象是更大的封装,根据职责在一个对象中封装多个方法
- 在完成某一个需求前,首先确定职责——要做的事情(方法)
- 根据职责确定不同的对象,在对象内部封装不同的方法(多个)
- 最后完成的代码,就是顺序地让不同的对象调用不同的方法
特点
- 注重对象和职责,不同的对象承担不同的职责
- 更加适合应对复杂的需求变化,是专门应对复杂项目开发,提供的固定套路
- 需要在面向过程基础上,在学习一些面向对象的语法
类和对象
类和对象的概念
类和对象是面向对象编程的两个核心概念
1. 类
- 类是一群具有相同特征或者行为的事物的一个统称,是抽象的,不能直接使用
- 特征被称为属性
- 行为被称为方法
- 类就相当于制造飞机时的图纸,是一个模板,是负责创建对象的
2.对象
-
对象是由类创建出来的一个具体存在,可以直接使用
-
由哪一个类创建出来的对象就拥有在哪一个类中定义的:
- 属性
- 方法
-
对象就相当于图纸制造的飞机
在程序开发中,应该先有类,再有对象
类和对象的关系
- 类是模板,对象是根据这个模板创建出来的,应该先有类,再有对象
- 类只有一个,而对象可以有很多个
- 不同的对象之间属性可能会各有不同
- 类中定义了什么属性和方法,对象中就有什么属性和方法,不可能多,也不可能少
类的设计
在使用面向对象开发前,应该首先分析需求,确定一下,程序中需要包含哪些类!
在程序开发中,要设计一个类,通常需要满足以下三个要素:
- 类名,这类事物的名字,满足大驼峰命名法
- 属性,这类事物具有什么样的特征
- 方法,这类事物具有什么样的行为
大驼峰命名法
CapWords
-
每一个单词的首字母大写
-
单词与单词之间没有下划线
类名的确定
名词提炼法 分析 整个业务流程,出现的 名词,通常就是找到的类
属性和方法的确定
- 对 对象的特征描述,通常可以定义成 属性
- 对象具有的行为(动词),通常可以定义成方法
提示:需求中没有涉及的属性或者方法在设计类时,不需要考虑
面向对象的基础语法
- 在
python
中对象几乎是无所不在的,我们之前学习的变量、数据、函数,都是对象
在Python
中可以使用以下两个方法验证一个家伙到底是不是对象:
- 在标识符/数据后输入一个
.
,然后按下TAB
键,ipython
会提示该对象能够调用的方法列表 - 使用内置函数
dir
传入标识符/数据,可以查看对象内的所有属性及方法
提示:_方法名_
格式的方法是python
提供的内置方法/属性
In [1]: num_list = []
In [2]: num_list.
append() count() insert() reverse()
clear() extend() pop() sort()
copy() index() remove()
In [3]: def demo():
...: """这是一个测试函数"""
...: print("hello world")
...:
In [4]: demo()
hello world
In [5]: dir(demo)
Out[5]:
['__annotations__',
'__call__',
'__class__',
'__closure__',
'__code__',
'......']
序号 | 方法名 | 类型 | 作用 |
---|---|---|---|
01 | __new__ | 方法 | 创建对象时,会被自动调用 |
02 | __init__ | 方法 | 对象被初始化时,会被自动调用 |
03 | __del__ | 方法 | 对象被从内存中销毁前,会被自动调用 |
04 | __str__ | 方法 | 返回对象的描述信息,print 函数输出使用 |
提示 利用好dir()
函数,在学习很多内容就不需要死记硬背了
定义简单的类
面向对象是更大的封装,在一个类中封装多个方法,这样通过这个类创建出来的对象,就可以直接调用这些方法了
定义只包含方法的类
- 在
python
中要定义一个只包含方法的类,语法格式如下:
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.eat()
tom.drink()
引用概念的强调
在面向对象开发中,引用的概念是同样适用的
- 在
python
中使用类创建对象之后,tom
变量中仍然记录的是对象在内存中的地址 - 也就是
tom
变量引用了新建的猫对象 - 使用
print
输出对象变量,默认情况下,是能够输出这个变量引用的对象是由哪一个类创建的对象,以及在内存中的地址(十六进制表示)
提示:在计算机中,通常使用十六进制来表示内存地址
- 十进制和十六进制都是用来表达数字的,只是表示的方式不一样
- 十进制和十六进制的数字之间可以来回转换
%d
可以以十进制输出数字%x
可以以十六进制输出数字
案例进阶 —— 使用 Cat
类再创建一个对象
class Cat:
"""这是一个猫类"""
def eat(self):
print("小猫爱吃鱼")
def drink(self):
print("小猫要喝水")
tom = Cat()
tom.eat()
tom.drink()
# 再创建一个对象
lazy_cat = Cat()
lazy_cat.eat()
lazy_cat.drink()
- 对象
tom
和对象lazy_cat
不是同一个对象- 类只有一个,但是可以使用同一个类创建出不同的对象
方法中的 self 参数
- 在
python
中,要给对象设置属性,非常的容易,但是不推荐使用- 因为:对象属性的封装应该封装在类的内部
- 只需要在 类的外部的代码中直接通过
.
设置一个属性即可
注意: 这种方式虽然简单,但是不推荐使用!
tom.name = "Tom"
...
lazy.name = "小懒猫"
使用 self 在方法内部输出每一只猫的名字
由 哪一个对象 调用的方法,方法内的
self
就是哪一个对象的引用
- 在类封装的方法内部,
self
就表示 当前调用发法的对象自己 - 调用方法时,程序员不需要传递
self
参数 - 在方法内部
- 可以通过
self.
访问对象的属性 - 也可以通过
self.
调用其他的对象方法
- 可以通过
- 改造代码如下:
class Cat:
"""这是一个猫类"""
def eat(self):
# 哪一个对象调用方法,self 就是哪一个对象的引用
print("%s 爱吃鱼" % self.name)
def drink(self):
print("%s 要喝水" % self.name)
tom = Cat()
tom.name = "Tom"
tom.eat()
tom.drink()
# 再创建一个对象
lazy_cat = Cat()
lazy_cat.name = "小懒猫"
lazy_cat.eat()
lazy_cat.drink()
初始化方法
- 在日常开发中,不推荐在类的外部给对象增加属性
- 如果在运行时,没有找到属性,程序会报错
- 对象应该包含哪些属性,应该封装在类的内部
初始化方法
-
当使用
类名()
创建对象时,会自动执行以下操作:- 为对象在内存中分配空间——创建对象
- 为对象的属性设置初始值——初始化方法
(init)
-
这个初始化方法就是
__init__
方法,__init__
是对象的内置方法
__init__
方法是专门用来定义一个类 具有哪些属性的方法
在 Cat
中增加 __init__
方法,验证该方法在创建对象时被自动调用
class Cat:
def __init__(self):
print("这是一个初始化方法")
# 使用类名()创建对象的时候,会自动帮我们调用初始化方法 __init__
tom = Cat()
在初始化方法内部定义属性
- 在
__init__
方法内部使用self.属性名 = 属性的初始值
就可以定义属性 - 定义属性之后,再使用
Cat
类创建的对象,都会拥有该属性
class Cat:
def __init__(self):
print("这是一个初始化方法")
self.name = "Tom"
def eat(self):
print("%s 爱吃鱼" % self.name)
tom = Cat()
tom.eat()
改造初始化方法——初始化的同时设置初始值
-
再开发中,如果希望在创建对象的同时,就设置对象的属性,可以对
__init__
方法进行改造- 把希望设置的属性值,定义成
__init__
方法的参数 - 在方法内部使用
self.属性 = 形参
接收外部传递的参数 - 在创建对象时,使用
类名(属性1,属性2...)
调用
class Cat: def __init__(self, name): print("这是一个初始化方法") self.name = name def eat(self): print("%s 爱吃鱼" % self.name) tom = Cat("Tom") tom.eat() lazy_cat = Cat("懒猫") lazy_cat.eat()
- 把希望设置的属性值,定义成
内置方法和属性
序号 | 方法名 | 类型 | 作用 |
---|---|---|---|
01 | __del__ | 方法 | 对象被从内存中销毁前,会被自动调用 |
02 | __str__ | 方法 | 返回对象的描述信息,print 函数输出使用 |
__del__方法
- 在
python
中- 当使用
类名()
创建对象时,为对象分配完空间后,自动调用__init__
方法 - 当一个对象被从内存中销毁前,会自动调用
__del__
方法
- 当使用
- 应用场景
__init__
改造初始化方法,可以让创建对象更灵活__del__
如果希望在对象被销毁前,再做一些事情,可以考虑一下__del__
方法
- 生命周期
- 一个对象从调用
类名()
创建,生命周期开始 - 一个对象的
__del__
方法一旦被调用,生命周期结束 - 在对象的生命周期内,可以访问对象属性,或者让对象调用方法
- 一个对象从调用
class Cat:
def __init__(self, name):
self.name = name
print("%s 来了" % self.name)
def __del__(self):
'''对象在内存中消失之前会自动调用此方法'''
print("%s 走了" % self.name)
tom = Cat("Tom")
__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):
"""必须返回一个字符串"""
return "对象的变量[%s]" % self.name
tom = Cat("Tom")
print(tom)