Python | 面向对象编程

本文总结 Python 面向对象编程相关的知识点 。

Updated: 2022 / 7 / 23



类和实例

概念

面向对象最重要的概念就是 类(Class)实例(Instance),必须牢记类是抽象的模板,比如 Student 类,而实例是根据类创建出来的一个个具体的 对象,每个对象都拥有相同的方法,但各自的数据可能不同。

打个比方,类是做菜的Reciple,而实例是根据Reciple做出来的菜(对象)。每个菜(对象)都拥有相同的方法,但各自的调料剂量(数据)可能不同。

仍以 Student 类为例,在 Python 中,定义类是通过 class 关键字。class 后面紧接着是类名,即 Student,类名通常是大写开头的单词,紧接着是 object,表示该类是从哪个类继承下来的,继承的概念我们后面再讲,通常,如果没有合适的继承类,就使用 object 类,这是所有类最终都会继承的类。如下所示:

class Student(object):
    pass

定义好了 Student 类,就可以根据 Student 类创建出 Student 的实例,创建实例是通过 类名+() 实现的。可以看到,变量 bart 指向的就是一个 Student 的实例,后面的 0x10a67a590 是内存地址,每个 object 的地址都不一样,而 Student 本身则是一个类。

>>> bart = Student()
>>> bart
<__main__.Student object at 0x10a67a590>
>>> Student
<class '__main__.Student'>

可以自由地给一个实例变量绑定属性,比如,给实例 bart 绑定一个 name 属性:

>>> bart.name = 'Bart Simpson'
>>> bart.name
'Bart Simpson'

由于类可以起到模板的作用,因此,可以在创建实例的时候,把一些我们认为必须绑定的属性强制填写进去。通过定义一个特殊的 __init__ 方法,在创建实例的时候,就把namescore 等属性绑上去:

class Student(object):

    def __init__(self, name, score):
        self.name = name
        self.score = score

注意到 __init__ 方法的第一个参数永远是 self,表示创建的实例本身,因此,在 __init__ 方法内部,就可以把各种属性绑定到 self,因为 self 就指向创建的实例本身。

有了 __init__ 方法,在创建实例的时候,就不能传入空的参数了,必须传入与 __init__ 方法匹配的参数,但 self 不需要传,Python 解释器自己会把实例变量传进去:

>>> bart = Student('Bart Simpson', 59)
>>> bart.name
'Bart Simpson'
>>> bart.score
59

和普通的函数相比,在类中定义的函数只有一点不同,就是第一个参数永远是实例变量self,并且,调用时,不用传递该参数。

除此之外,类的方法和普通函数没有什么区别,所以,你仍然可以用默认参数、可变参数、关键字参数和命名关键字参数。


数据封装

面向对象编程的一个重要特点就是数据封装。
在上面的 Student 类中,每个实例就拥有各自的 namescore 这些数据。我们可以通过函数来访问这些数据,比如打印一个学生的成绩:

>>> def print_score(std):
...     print('%s: %s' % (std.name, std.score))
...
>>> print_score(bart)
Bart Simpson: 59

但是,既然 Student 实例本身就拥有这些数据,要访问这些数据,就没有必要从外面的函数去访问,可以直接在 Student 类的内部定义访问数据的函数,这样,就把 “数据” 给封装起来了。这些封装数据的函数是和Student类本身是关联起来的,我们称之为类的方法:

class Student(object):

    def __init__(self, name, score):
        self.name = name
        self.score = score

    def print_score(self):
        print('%s: %s' % (self.name, self.score))

要定义一个方法,除了第一个参数是 self 外,其他和普通函数一样。要调用一个方法,只需要在实例变量上直接调用,除了 self 不用传递,其他参数正常传入:

>>> bart.print_score()
Bart Simpson: 59

这样一来,我们从外部看 Student 类,就只需要知道,创建实例需要给出 namescore,而如何打印,都是在 Student 类的内部定义的,这些数据和逻辑被“封装”起来了,调用很容易,但却不用知道内部实现的细节。

封装的另一个好处是可以给 Student 类增加新的方法,比如 get_grade

class Student(object):
    ...

    def get_grade(self):
        if self.score >= 90:
            return 'A'
        elif self.score >= 60:
            return 'B'
        else:
            return 'C'

同样的,get_grade 方法可以直接在实例变量上调用,不需要知道内部实现细节:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

    def get_grade(self):
        if self.score >= 90:
            return 'A'
        elif self.score >= 60:
            return 'B'
        else:
            return 'C'

小结

类是创建实例的模板,而实例则是一个一个具体的对象,各个实例拥有的数据都互相独立,互不影响;

方法就是与实例绑定的函数,和普通函数不同,方法可以直接访问实例的数据;

通过在实例上调用方法,我们就直接操作了对象内部的数据,但无需知道方法内部的实现细节。

和静态语言不同,Python 允许对实例变量绑定任何数据,也就是说,对于两个实例变量,虽然它们都是同一个类的不同实例,但拥有的变量名称都可能不同:

>>> bart = Student('Bart Simpson', 59)
>>> lisa = Student('Lisa Simpson', 87)
>>> bart.age = 8
>>> bart.age
8
>>> lisa.age
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'age'

对象

参考这里 1


获取信息

当我们拿到一个对象的引用时,如何知道这个对象是什么类型、有哪些方法呢?
首先,我们来判断对象类型,使用 type() 函数,基本类型都可以用 type() 判断:


方法

type

type() 不会认为子类是一种父类类型,不考虑继承关系。


用法
  • type(name, bases, dict)
  • name 是类的名称;
  • bases 是基类的元组;
  • dict 是类内定义的命名空间变量;

当有一个参数时它的返回值是对象类型,。
有三个参数时,返回值是新的类型对象。


示例
  1. 基本类型
type(123)
# <class 'int'>

type('str')
# <class 'str'>

type(None)
# <class 'NoneType'>
  1. 函数或者类
type(abs)
# <class 'builtin_function_or_method'>

type(a)
# <class '__main__.Animal'>

但是 type() 函数返回的是什么类型呢?它返回对应的 Class 类型。
如果我们要在 if 语句中判断,就需要比较两个变量的 type 类型是否相同:

type(123) == type(456)
# True
type('abc') == type('123')
# True
type('abc')==type(123)
# False

type(123)==int
# True
type('abc')==str
# True

判断基本数据类型可以直接写 intstr 等,但如果要判断一个对象是否是函数怎么办?可以使用 types 模块中定义的常量。


types
import types

type(abs)==types.BuiltinFunctionType
True

def fn():
    pass
type(fn)==types.FunctionType
# True

type(lambda x: x)==types.LambdaType
# True

type((x for x in range(10)))==types.GeneratorType
# True

isinstance

instance 会认为子类是一种父类类型,考虑继承关系。
如果要判断两个类型是否相同,使用 isinstance() 更好,而不是 type()。一般 type(x) 用来看变量 x 的类型较多 2

用法
  • isinstance(object, classinfo)
  • object 是实例对象,变量
  • classinfo 可以是直接或间接类名、基本类型或者由它们组成的元组(如tuple, dict, int, str, float, list, set, bool, class 类等)。
    如果对象的类型与 classinfo 相同则返回值为 True,否则返回值为 False

示例

如果是继承关系,

object -> Animal -> Dog -> Husky

那么,isinstance() 就可以告诉我们,一个对象是否属于某种类型,先创建3种类型的对象:

a = Animal()
d = Dog()
h = Husky()

然后,判断:

isinstance(h, Husky)
# True
# 因为 h 变量指向的就是 Husky 对象。

再判断:

isinstance(h, Dog)
# True
# 虽然 h 自身是 Husky 类型,但由于 Husky 是从 Dog 继承下来的。所以,h 也还是 Dog类型。换句话说,isinstance()判断的呀对象是否是该类型本身,或者位于该类型的父继承链上。

因此,我们可以确信,hAnimal 类型:

isinstance(h, Animal)
# True

# 同理,实际类型是 Dog 的 d 也是 Animal 类型
isinstance(d, Dog) and isinstance(d, Animal)
# True

但是,d 不是 Husky 类型:

instance(d, Husky)
# False

能用 type() 判断的基本类型也可以用 isinstance() 判断:

isinstance('a', str)
# True

isinstance(123, int)
# True

isinstance(b'a', bytes)
# True

并且还可以判断一个变量是否是某些类型中的一种,比如下面的代码就可以判断是否是 list 或者 tuple

isinstance([1, 2, 3], (list, tuple))
# True

isinstance([1, 2, 3], (list, tuple)
# True)

总是优先使用 isinstance() 判断类型,可以将指定类型及其子类 ‘一网打尽`。


dir

如果要获得一个对象的所有属性和方法,可以使用 dir() 函数,它返回一个包含字符串的 list,比如,获得一个 str 对象的所有属性和方法:

dir('ABC')
# ['__add__', '__class__',..., '__subclasshook__', 'capitalize', 'casefold',..., 'zfill']
# 类似 __xxx__ 的属性和方法在Python中都是有特殊用途的,比如 __len__ 方法返回长度。在 Python 中,如果你调用 len() 函数试图获取一个对象的长度,实际上,载 len() 函数内部,它自动去调用该对象的 __len__() 方法,所以,下面的代码是等价的:len('ABC'), 'ABC'.__len__()

我们自己写的类,如果也想用 len(myObj) 的话,就自己写一个 __len__() 方法:

class MyDog(object):
    def __len__(self):
        return 100


dog = MyDog()
print(len(dog))
# 100

剩下的都是普通属性或方法,比如 lower() 返回小写的字符串:

'ABC'.lower()
# abc

仅仅把属性和方法列出来是不够的,配合 getattr()setattr() 以及 hasattr(),我们可以直接操作一个对象的状态:


属性
class MyObject(object):
    def __init__(self):
        self.x = 9
    def power(self):
        return self.x * self.x

obj = MyObject()

###############################

hasattr(obj, 'x')
# 有属性 `x` 吗?
# True
# 该对象具有 `x` 属性

obj.x
# 9
# 该对象的 `x` 属性的值为 9
obj.power()
# 81
# 该对象的 power 方法的返回值为 81

###############################

hasattr(obj, 'y')
# 有属性'y'吗?
# False
# 该对象不具有 `y` 属性

setattr(obj, 'y', 19)
# 对该对象设置 `y` 属性

hasattr(obj, 'y')
# 有属性'y'吗?
# True
# 该对象具有 `y` 属性

getattr(obj, 'y')
# 获取属性'y'
# 19

###############################

getattr(obj, 'z')
# 如果试图获取不存在的属性,会抛出 `AttributeError` 的错误:
# 获取本来不存在的属性 `z`
# Traceback (most recent call last):
#   File "/Users/xueshanzhang/PycharmProjects/pythonProject0312/test.py", line 43, in <module>
#     print(getattr(obj, 'z'))
# AttributeError: 'MyObject' object has no attribute 'z'

getattr(obj, 'z', 404)
# 在获取属性的值的时候,如果不存在该属性,则返回一个默认值,即 `404`
# 404

方法
class MyObject(object):
    def __init__(self):
        self.x = 9
    def power(self):
        return self.x * self.x

obj = MyObject()

hasattr(obj, 'power')
# 有 power 属性吗?
# True

getattr(obj, 'power')
# 获取属性 power
# <bound method MyObject.power of <__main__.MyObject object at 0x10149cc10>>

fn = getattr(obj, 'power')
# 获取属性 power 并赋值到变量 fn
# fn指向 obj.pow
# <bound method MyObject.power of <__main__.MyObject object at 0x102be0c10>>er

fn()
# 81
# 调用 fn() 和 调用 obj.power() 是一样的效果

小结

通过内置的一系列函数,我们可以对任意一个 Python 对象进行剖析,拿到其内部数据。
要注意地是,只有在不知道对象信息的时候,我们才会去获取对象信息。

如果可以直接写

sum = obj.x + obj.y

就不要写:

sum = getattr(obj, 'x') + getattr(obj, 'y')

一个正确的用法的例子如下:

'''
假设我们希望从文件流 fp 中读取图像,我们首先可以使用 hasattr() 来判断该 fp 对象是否存在 read 方法:
如果存在,则该对象是一个 流;
如果不存在,则无法读取;

请注意,在 Python 这类动态语言中,根据类型,有 read() 方法,不代表该 fp 对象就是一个文件流,也可能是网络流,
也可能是内存中的一个字节流,但只要 read() 方法返回的是有效的图像数据,就不影响读取图像的功能。
'''

def readImage(fp):
    # 判断 fp 对象是否存在 read 方法
    if hasattr(fp, 'read'):
        # fp 对象存在 read 方法,fp 对象是一个流
        return readData(fp)
        # 返回读取到的图像内容
    return None

实例及类

参考这里 1


属性

由于 python 是动态语言,根据类创建的实例可以任意绑定属性。
给实例绑定属性的方法是通过实例变量,或者通过 self 变量:

class Student(object):
    def __init__(self, name):
        self.name = name

s = Student('Bob')
s.score = 90

但是,如果 Student 类本身需要绑定一个属性呢?
可以直接在 class 中定义属性,这种属性是类属性,归 Student 类所有:

class Student(object):
	name = 'Student'

当我们定义了一个类属性后,这个属性虽然归类所有,但类的所有实例都可以访问到。

'''
在编写程序时,不建议对实例属性和类属性使用相同的名字,因为相同名称的实例属性将屏蔽掉类属性。
但是当删除实例属性侯,若再使用相同名称,将访问到类的同名属性。
'''

class Student(object):
    name = 'Student'

s = Student()
# 创建实例

s.name
# Student
# 打印 name 属性。但是因为实例并没有 name 属性,所以会继续查找class的name属性
Student.name
# Student
# 打印类的 name 属性

s.name = 'Michael'
# 对实例绑定 name 属性
s.name
# Michael
# 由于实例属性优先级比类属性高,因此,它会屏蔽掉类的 name属性
Student.name
# Student
# 即使给实例s绑定name属性,但类的name属性并未消失,因此还可以使用Student.name进行访问

del s.name
# 使用 del 来删除实例的name属性
s.name
# 再次调用s.name。由于实例的name属性并未寻找到,因为调用类的name属性。

小结

实例属性属于各个实例所有,互不干扰;
类属性属于类所有,所有实例共享一个属性;
不要对实例属性和类属性使用相同的名字,否则将产生难以发现的错误。


参考链接


  1. Python爬虫技术–基础篇–面向对象编程(下) ↩︎ ↩︎

  2. Python判断 NoneType数据类型 ↩︎

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值