11. 面向对象,类的概念,类的特殊方法
1. 面向对象
1. 面向对象的概念
- Python是一门面向对象的编程语言
- 所谓面向对象的语言,简单理解就是语言中的所有操作都是通过对象来进行的
2. 面向过程与面向对象的比较
- 面向过程
• 面向过程指将我们的程序分解为一个一个步骤,通过对每个步骤的抽象来完成程序
• 这种编写方式往往只适用于一个功能,如果要实现别的功能,往往复用性比较低
• 这种编程方式符号人类的思维,编写起来比较容易 - 面向对象
• 面向对象的编程语言,关注的是对象,而不注重过程,对于面向对象一切皆对象
• 面向对象的编程思想,将所有功能统一保存到对应的对象中,要使用某个功能,直接找到对应的对象即可
• 这种编码方式比较容易阅读,并且易于维护,容易复用。但是编写的过程中不太符合常规的思维,编写相对麻烦 - 比较
这里使用一个例子:解决将大象装进冰箱这个问题。
面向过程 执行者
按照步骤,一步步执行
重在于操作本身
1:打开冰箱门 等各种操作
2:装大象 等各种操作
3:关冰箱门 等各种操作
面向对象 指挥者
建立一个类“Refrigerator”,
包含有“open”(开门),“store”(储存),“close”(关门)
三种针对对象“Refrigerator”的方法
按照“Refrigerator”这个类,建立一个实例“Refrigerator1”
然后另一个对象“Elephant”被引入,
调用出该实例“Refrigerator1”的三种方法,对对象“Elephant”进行处理
1. Refrigerator1.open
2.Refrigerator1.store
3. Refrigerator1.close
- 面向对象的优势:
• 面向对象的编程思想,将所有功能统一保存到对应的对象中,要使用某个功能,直接找到对应的对象即可
• 这种编码方式比较容易阅读,并且易于维护,容易复用。但是编写的过程中不太符合常规的思维,编写相对麻烦
2. 类(class)
1. 类的概念
• 我们目前学习的都是Python的内置对象,但是内置对象并不都能满足我们的需求,所以我们在开发中经常要自定义一些对象
• 类简单理解它就是相当于一个图纸,在程序汇总我们需要根据类来创建对象;而类就是对象的图纸。我们也称对象是类的实例(instance)
• 如果多个对象是通过一个类创建的,我们称这些对象是一类对象
2. 定义类对象
语法
class 类名([父类]):
pass
3. 类的定义
• 用来描述具有相同的属性和方法的对象的集合。它定义了该集合中每个对象所共有的属性和方法。对象是类的实例。
• 类也是一个对象 ,类就是用来创建对象的对象
4. 类的特征
• 类和对象都都包含两部分:变量(属性),行为(方法)
• 对于类:
- 类变量(属性)
• 可以向对象中添加变量,而对象中的变量称之为属性
• 类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体(方法)之外。类变量通常不作为实例变量使用,类变量也称作属性
• 语法:
属性值 = 对象.属性名
- 行为(方法)
• 方法:类中定义的函数。
• 方法调用和函数调用的区别:如果是函数调用,调用时有几个形参,就会传递几个实参。如果是方法调用,默认传递一个参数,所以方法中至少得有一个形参
• 调用方法:
对象.方法名()
- 注意:
在类代码块中,我们可以定义变量和函数
• 变量会成为该类实例的公共属性,所有的该实例都可以通过 对象.属性名的形式访问
• 函数会成为该类实例的公共方法,所有该类实例都可以通过 ==对象.方法名()==的形式访问
class MyClass(): # 定义类 图纸
color = 'red' # 属性
def run(self): # 行为(方法)
print('车子迅疾的跑起来了')
mc = MyClass()
# 创建类的实例对象(类的实例化) 通过图纸创造出来的实在的东西
print(mc.color)
mc.run()
此处
class MyClass():就是在创造类
color就是其类变量(属性)
def run(self)就是在定义行为(方法)
mc就是根据MyClass创造出来的实例
mc.color就是查看其属性
mc.run()就是在调用其内置方法
- 属性和方法
• 类中定义的属性和方法都是公共的,任何该类实例都可以访问
• 属性和方法的查找流程
• 当我们调用一个对象的属性时,解析器判断当前的对象中寻找是否还有该属性,如果有,则直接返回当前的对象的属性值;如果没有,则去当前对象的类对象中去寻找,如果有则返回类对象的属性值。如果没有就报错
• 类对象和实例对象中都可以保存属性(方法)
class Person():
pass
person1 = Person()
person1.name = 'skx'
print(person1.name)
>skx
- 注意:
• 如果这个属性(方法)是所以的实例共享的,则应该将其保存到类对象中
• 如果这个属性(方法)是摸个实例独有的。则应该保存到实例对象中
• 一般情况下,属性保存到实例对象中 而方法需要保存到类对象中
- self参数
• self在定义时需要定义,但是在调用时会自动传入。
• self的名字并不是规定死的,但是最好还是按照约定是用self
• self总是指调用时的类的实例
class Person():
def run(self):
print(self)
p1 = Person()
print(p1)
p1.run()
><__main__.Person object at 0x000002448B1842E8>
><__main__.Person object at 0x000002448B1842E8>
5.类的特殊方法
• 在类中可以定义一些特殊方法也称为魔术方法
• 特殊方法都是形如 xxx()这种形式
• 特殊方法不需要我们调用,特殊方法会在特定时候自动调用
如:原本定义一个类的的属性和方法是这样做的
class Person():
name = ' '
def speak(self): # 谁调用speak方法,self就是那个对象本身
print('大家新年好, 我是%s' % self.name)
p1 = Person()
p1.name = 'skx'
print(p1.name)
p1.speak()
>skx
>大家新年好,我是skx
引入魔法方法__init__()以后,该函数可以自动在创建实例的时候就被调用
class Person():
def __init__(self, name): # 对于init方法,实例创建的时候就被调用了
self.name = name
def speak(self): # 谁调用speak方法,w就是那个对象本身
print('大家新年好, 我是%s' % self.name)
p1 = Person('skx')
print(p1.name)
p1.speak()
>skx
>大家新年好,我是skx
因为
name这个属性是必须的
name这个属性又各不相同
手动添加name属性容易遗忘,假使没有忘记,查看源码非常的麻烦
因此很明显,后者这种用法更加简便
而
两种用法的区别在于
前者的name属性是在定义类时就定义了,是这个类所有的实例都共有的
后者的name是在建立各个实例时才赋予的,是根据不同实例定制的
• 常用特殊方法
init : 构造函数,在生成对象时调用
del : 析构函数,释放对象时使用
repr : 打印,转换
setitem : 按照索引赋值
getitem: 按照索引获取值
len: 获得长度
cmp: 比较运算
call: 函数调用
add: 加运算
sub: 减运算
mul: 乘运算
truediv: 除运算
mod: 求余运算
pow: 乘方