Skr-Eric的Python课堂(二十一)——Python的类的延伸讲解(续)

用于类的函数

  issubclass(cls, class_or_tuple) 判断一个类是否继承自其它的类,如果此类cls是class或tuple中一个派生子类则返回True,否则返回False

 

  示例:

class A:
    pass

class B(A):
    pass

class C(B):
    pass

issubclass(C, B)  # True
issubclass(B, A)  # True
issublcass(A, C)  # False

 

封装 enclosure

  封装是指隐藏类的实现细节,让使用者不关心这些细节

  封装的目的是让使用者通过尽可能少的方法(或属性)操作对象

 

私有属性和方法

  python类中,以双下划线('__') 开头,不以双下划线结尾的标识符为私有成员,私有成员只能使用该类的方法来进行访问和修改

    1. 以__开头的属性为私有属性

    2. 以__开头的方法为私有方法

示例:

# 此示例示意用私有属性和私有方法来实现封装
class A:
    def __init__(self):
        self.__p1 = 100  # <<--- __p1为私有属性
        # self.__p2__ = 200  # 这人不是私有属性

    def show_info(self):
        print(self.__p1)  #此对象的实例方法可以访问和修改私有属性
        self.__m()  # 调用私有方法

    def __m(self):
        print("A类对象的__m方法被调用")

a = A()
a.show_info()  # 100
# a.__m()  # 出错, 除A类的实例方法外,不能调用a对象的私有方法
# print(a.__p1)  # 不允许访问私有属性
# print(a.__p2__)

 

  注:

    python 的封装是假的封装(模拟的封装)

 

多态 polymorphic

  字面意思: '多种状态'

  多态是指在有继承/派生关系的类中,调用基类对象的方法,实际能调用子类的覆盖方法的现象叫多态

 

  状态:

    静态(编译时状态)

    动态(运行时状态)

  说明:

    1. 多态调用的方法与对象相关,不写类型相关

    2. Python全部对象都只有"运行时状态(动态)", 没有"C++语言"里的编译时状态(静态)

 

  示例:

class Shape:
    def draw(self):
        print("Shape的draw()被调用")

class Point(Shape):
    def draw(self):
        print('正在画一个点!')

class Circle(Point):
    def draw(self):
        print('正在画一个圆!!!')

def my_draw(s):
    s.draw()  # <<<--- 此处显示出多态中的"动态"

s1 = Circle()  # 创建一个圆对象
s2 = Point()  # 创建一个点对象
my_draw(s1)
my_draw(s2)

 

面向对象编程语言的特征:

  继承

  封装

  多态

 

多继承 multiple inheritance

  多继承是指一个子类继承自两个或两个以上的基类

 

  语法:

    class 类名(基类名, 基类名2, .....):

        pass

  说明:

    一个子类同时继承自多个父类,父类中的方法可以同时被继承下来

    如果两个父类中有同名的方法,而在子类中又没有覆盖此方法时,调用结果难以确定

  示例:

# 此示例示意多继承的定义语法的基本用法
class Car:
    '''汽车类'''
    def run(self, speed):
        print("汽车以", speed, '公里/小时的速度行驶')

class Plane:
    '''飞机类'''
    def fly(self, height):
        print("飞机以海拔", height, '米的高度飞行')

class PlaneCar(Car, Plane):
    '''PlaneCar类同时继承自汽车类和飞机类'''

p = PlaneCar()  # 创建一个飞行汽车对象
p.fly(10000)
p.run(300)

 

多继承的问题(缺陷)

  标识符冲空问题

    (要谨慎使用多继承)

 

  示例:

# 小张写了一个类A:
class A:
    pass
    # def m(self):
    #     print("A.m() 被调用")

# 小李写了一个类B:
class B:
    def m(self):
        print("B.m() 被调用")

# 小王感觉小张和小李写的两个类自己都可以用
class AB(A, B):
    pass
    # def m(self):
    #     print("AB.m() 被调用")

ab = AB()
ab.m()  # 请问调用谁? 为什么?

 

多继承的 MRO(Method Resolution Order) 问题

  python3的类的__mro__属性

    作用:  用来记录类的方法查找顺序

 

   示例:

class A:
    def go(self):
        print("A")

class B(A):
    def go(self):
        print("B")
        super().go()  # C

class C(A):
    def go(self):
        print("C")

class D(B, C):
    def go(self):
        print("D")
        super().go()  # 调用谁?

d = D()
d.go()

 

super() 函数就是根据__mro__来调用上层的方法

 

函数重写 overwrite

  什么是函数重写

    在自定义的类内添加相应的方法,让自定义的类创建的实例能够使用内建函数进行操作

  

对象转字符串函数

  repr(x) 返回一个能表示python对象的表达式字符串,通常

     eval(repr(obj)) == obj

  str(x)  通过给定的对象返回一个字符串(这个字符串通常供人阅读)

 

  示例:

s = "I'm a teacher"
print(str(s))  # I'm a teacher
print(repr(s))  # "I'm a teacher"

 

对象转字符串函数的重写方法:

  repr() 函数的重写方法:

      def __repr__(self):

          return 字符串

 

  str() 函数的重写方法:

      def __str__(self):

          return 字符串

 

  str(obj) 函数调用方法说明:

    1. str(obj) 函数先查找obj.__str__(方法), 调用此方法并返回结果

    2. 如果 obj.__str__() 方法不存在.则调用obj.__repr__方法并返回结果

    3. 如果 obj.__repr__方法不存在,则调用 object类的__repr__实例方法显示<__main__.XXXX object at 0xXXXXXXX> 格式的字符串

  示例:

# 此示例示意对象转字符串函数的重写方法
class MyNumber:
    def __init__(self, val):
        self.data = val  # 在每个对象内部都创建一个实例变量来绑定数据

    def __str__(self):
        # print("__str__方法被调用")
        return "自定义数字: %d" % self.data

    def __repr__(self):
        '''此方法返回来的字符串一定是能表示self对象的表达式字符串'''
        return "MyNumber(%d)" % self.data

n1 = MyNumber(100)
print('str(n1) =', str(n1))  # 自定的数字:100
print('repr(n1) =', repr(n1))  # MyNumber(100)

n2 = MyNumber(200)
print(str(n2))
print(n2.__str__())
print(n2)  # 在print内部会将n2用str(x) 转为字符串再写到sys.stdout

 

内建函数重写

  方法名                         函数名

 def __abs__(self):         abs(obj)  函数调用

 def __len__(self):         len(obj)  函数调用

 def __reversed__(self):    reversed(obj)  函数调用

 def __round__(self):       round(obj)  函数调用

 

示例:

class MyList:
    '''这是一个自定义的列表类型,
    此类型的对象用data属性绑定的列表来存储数据'''
    def __init__(self, iterable=()):
        self.data = [x for x in iterable]

    def __repr__(self):
        return 'MyList(%s)' % self.data

    def __len__(self):
        return len(self.data)

    def __abs__(self):
        L = [abs(x) for x in self.data]
        return MyList(L)

myl = MyList([1, -2, 3, -4])
print(myl)  # MyList([1, -2, 3, -4])
print(len(myl))   # 4
print(abs(myl))   # MyList([1, 2, 3, 4])

 

数值转换函数的重写:

  def __complex__(self):     complex(obj) 函数调用

  def __int__(self):        int(obj)

  def __float__(self):      float(obj)

  def __bool__(self):       bool(obj)

  示例:

# 此示例示意数值转换函数的重写
class MyNumber:
    def __init__(self, val):
        self.data = val

    def __repr__(self):
        return "MyNumber(%d)" % self.data

    def __int__(self):
        '''重写int(obj) 函数'''
        return int(self.data)

    def __float__(self):
        return float(self.data)

n1 = MyNumber(100)
n = int(n1)  # 出错
print(n)

f = float(n1)
print(f)

c = complex(n1)  # 当没有n1.__complex__() 时会调用n1.__float__() + 0j
print(c)

 

布尔测试函数的重写

  格式:

    def __bool__(self):

        ...

  作用:

    用于bool(obj) 函数取值

    用于if语句真值表达式中

    用于while语句的真值表达式中

  说明:

    1. 当自定义类内有__bool__(self) 方法时,此方法的返回作为bool(x)的返回值

    2. 当不存在__bool__(self) 方法时,返回__len__(self) 方法的返回值是否为非零来测式布尔值

    3. 当不存在__len__(self) 方法时,则直接返回True

 

对象的属性管理函数:

  getattr(obj, name[,default])  从一个对象用字符串name得到对象的属性,getattr(x, 'y')等同于x.y;当属性不存在时,如果给定default参数则返回default,如果没有给出default 则触发一个AttributeError错误

 

  hasattr(obj, name)  用给定的name字符串返回obj是否有此属性,此种做法可以避免在getattr(obj, name) 时引发错误

 

  setattr(obj, name, value), 给对象obj的名为name的属性设置相应的值value, setattr(x, 'y', v) 等同于 x.y = v

 

  delattr(obj, name) 删除对象obj中的name属性 del(x, 'y') 等同于 del x.y

 

迭代器(高级)

  什么是迭代器

    由iter(x) 函数返回,可以通过next(it) 函数取值的对象就是迭代器

 

  迭代器协议:

    迭代器协议是指对象能够使用next()函数获取下一项数据,在没有下一项数据时触发一个StopIteration异常来终止迭代的约定

 

  迭代器协议的实现方法:

    def __next__(self):

        ....

    注:此方法需要实现迭代器协议

 

  什么是可迭代对象

    是指能用iter(obj) 函数返回迭代器的对象(实例)

    可迭代对象内部要定义__iter__(self) 方法来返回迭代器对象

 

  示例:

# 此示例示意将自定义的类MyList创建的对象制作成为可迭代对象

class MyList:
    def __init__(self, iterable=()):
        self.data = [x for x in iterable]

    def __repr__(self):
        return 'MyList(%s)' % self.data

    def __iter__(self):
        '''此方法用于返回一个能访问self对象的迭代器'''
        print("__iter__被调用")
        return MyListIterator(self.data)  # 创建迭代器并返回

class MyListIterator:
    '''此类用来描述能够访问MyList类型的对象的迭代器'''
    def __init__(self, lst):
        self.data_lst = lst
        self.cur_index = 0   # 迭代器访问的起始位置

    def __next__(self):
        '''此方法用来实现迭代器协议'''
        print('__next__方法被调用')
        if self.cur_index >= len(self.data_lst):
            raise StopIteration

        r = self.data_lst[self.cur_index]
        self.cur_index += 1
        return r

myl = MyList([2, 3, 5, 7])
it = iter(myl)  # 等同于调用 it = myl.__iter__()
print(next(it))  # 2
# print(next(it))  # 3
# print(next(it))  # 5
# print(next(it))  # 7
# print(next(it))  # StopIteration

for x in myl:
    print(x)

L = [x**2 for x in myl]
print(L)

 

 

课后习题:

  1. 实现原学生信息管理系统的Student类的封装,让除Student实例方法外的函数或其它方法都不能访问姓名,年龄,成绩等属性

 

  2. 写一个实现迭代器协议的类,让此类可以生成从b 开始的n个素数

    class Prime:

        def __init__(self, b, n):

            ...

        def __iter__(self):

           ....

 

    L = [x for x in Prime(10, 4)]

    print(L)  # L = [11, 13, 17, 19]

 

  3. 写一个类Fibonacci实现迭代器协议,此类的对象可以作为可迭代对象生成斐波那契数列

         1 1 2 3 5 8 13 ....

    class Fibonacci:

        def __init__(self, n):

            ...

        ...

    for x in Fibonacci(10):

        print(x)  # 打印 1 1 2 3 5 8 ...

 

 

想要看更多的课程请微信关注SkrEric的编程课堂

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值