IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Python知识库 -> python学习--面向对象(高级) -> 正文阅读

[Python知识库]python学习--面向对象(高级)

属性访问函数

注意函数参数中的属性需要用""括起来。

class Person:
	height = 100
    
    def __init__(self):
        self.name = "ice"
        self.age = 19
        self.sex = "man"
  1. hasattr()

用于判断属性是否存在。

print(hasattr(ps, "name"))  # True
# 也可以用于类变量判断
print(hasattr(Person, "height"))  # True
  1. getattr

得到属性值,没有则报错。

name2 = getattr(ps, "name")
print(name2)  # ice
  1. setattr

设置属性值

setattr(ps, "name", "冰鸽")  # 有则改
print(ps.name)  # 冰鸽
setattr(ps, "height", 180)  # 无则增
print(ps.height)  # 180
  1. delattr

删除属性

delattr(ps, "height")
print(hasattr(ps, "height"))  # False
delattr(ps, "height")  # 本来就没有的会报错

所以我们可以使用hasattr()函数防止报错,这里就不做演示了。

对象的关系方法

1.子类判断

issubclass用来判断一个类是不是另一个类的子类。如下,Person类是object类的子类。

print(issubclass(Person, object))  # True

2.实例判断

isinstance用来判断一个对象是不是一个类的实例化对象。

ps = Person()
print(isinstance(ps, Person))  # True
# 也可用于判断数据类型,以下判断5是否为int或float型
print(5, (int, float))  # True

一些魔法方法

1.new函数

    def __new__(cls, *args, **kwargs):
        print("该方法在实例化之前自动执行")

由于方法在实例化前自动执行,所以该方法的第一个参数是cls,而不是self。并且调用new函数时将不会有实例化对象产生。当然也可以创建实例化对象,此时就需要调用object类的new方法:

    def __new__(cls, *args, **kwargs):
        print("该方法在实例化之前自动执行")
        return super().__new__(cls)

因为object类的new方法是可以创建实例对象,但注意需要使用return返回。

2.单例模式

无论进行多少次实例化,得到的对象永远是唯一的一个。结合所学的hasattr函数可以完成所述操作。

    def __new__(cls, *args, **kwargs):
        print("该方法在实例化之前自动执行")
        if not hasattr(cls, "instance") :
            cls.instance = super().__new__(cls)
        return cls.instance

解析:每次进行实例化前都会执行new函数返回instance变量,但是在第二次调用时其返回的instance变量是第一次的结果,所以每次实例化的对象总是同一个。

3.重写输出方法

使用str函数可以实现,重写输出方法。

    def __str__(self):
        return "这是重写的输出方法"


print(ps)  # 这是重写的输出方法

4.查看类的函数

使用dir函数可以查看到某个类的所有函数。

print(dir(Person))

在这里插入图片描述

其中若含有__iter__该方法则代表该类可以被迭代,显然Person类是不能被迭代的。

协议与合同

协议需要两个或以上的魔法方法组成。
可迭代对象有:列表、元组、字符串,而迭代协议就是魔法方法__iter__

1.序列协议

序列协议需要len方法

class Person:
    def __init__(self, *args):
        self.args = args

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


ps = Person(1, 4, 5, 2, 4, 7)
print(len(ps))  # 6

实际上len获取到的长度就是传入元组的长度。但是此时Person类还不能像列表那样进行下标取值。这需要签订getitem协议。

    def __getitem__(self, item):
        return self.args[item]


print(ps[0])  # 1

2.迭代器协议

上面已经提到了该协议需要__iter__方法。当然其实还需要__next__协议才能迭代。

li = [1, 2, 3]
f = li.__iter__()  # f是列表的迭代器
print(next(f))  # 1
print(next(f))  # 2
print(next(f))  # 3
print(next(f))  # 报错StopIteration

以上是列表的例子,学习他我们编写自己的协议。

class Number:
    def __init__(self, end):
        # 像列表的range一样需要终止值
        self.start = -1  # 起始值,默认是-1
        self.end = end  # 终止值

    def __iter__(self):
        return self

    def __next__(self):
        self.start += 1
        if self.start >= self.end:
            #  主动报错 结束,类似列表一样当对列表执行多次next而超过时也会报出该错误
            raise StopIteration
        return self.start


num = Number(10)
for i in num:
    print(i)

3.上下文协议

enter和exit方法是该协议需要的方法,enter方法在程序执行之前就会执行,而exit方法在程序结束的时候才会自动执行。并且程序需通过with触发上下问协议。

import time


class Runtime:
    def __init__(self):
        pass

    def __enter__(self):
        self.start = time.time()  # 该函数可以获取当前的时间
        return self.start

    def __exit__(self, exc_type, exc_val, exc_tb):
        self.end = time.time()
        self.run = self.end - self.start
        print(f"程序运行了{self.run}")


# 通过with触发上下问协议
with Runtime():
    # 先调用enter
    # 要执行的程序
    for i in range(10000):
        pass
    # 调用exit程序

# 程序运行了0.0010006427764892578

上次练习答案

# 1.定义一个账户类,可以创建账户、存款、取款 、查询余额、以及销户等操作。
# 2.现在三个人分别去开户,存款 和 销户,请利用上面的类实现出来。
class Account:
    def __init__(self):
        self.account = None
        self.password = None
        self.money = 0

    # 创建账户
    def creat_account(self, account, password):
        self.account = account
        self.password = password
        self.money = 0

    def deposit(self, money, password):
        if self.password == password:
            self.money += money

    def draw_money(self, money, password):
        if self.password == password and self.money >= money:
            self.money -= money

    def query(self, password):
        if self.password == password:
            return self.money

    def cancel(self, password):
        if self.password == password:
            return f"{self.account}被删除了"
            del self

    def __del__(self):
        pass

    def toString(self):
        return f"account = {self.account}, money = {self.money}."


a = Account()
b = Account()
c = Account()

a.creat_account(1, 1)
b.creat_account(2, 2)
c.creat_account(3, 3)
print(a.toString())
b.deposit(100, 2)
print(b.toString())
print(c.cancel(3))

练习

  1. 测试使用列表推导式和不使用列表推导式哪个速度更快。(尽量写大一点)
  2. range不可以使用小数做步长,实现一个可迭代对象,可以实现小数步长。

结束语

点赞!!!
ps:现在关注我,以后就是老粉啦!!!

下篇预告

在协议中介绍了迭代器协议,但是需要定义类,那么只用函数可以生成嘛?

  Python知识库 最新文章
Python中String模块
【Python】 14-CVS文件操作
python的panda库读写文件
使用Nordic的nrf52840实现蓝牙DFU过程
【Python学习记录】numpy数组用法整理
Python学习笔记
python字符串和列表
python如何从txt文件中解析出有效的数据
Python编程从入门到实践自学/3.1-3.2
python变量
上一篇文章           查看所有文章
加:2022-03-06 12:58:23  更:2022-03-06 13:02:36 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/15 22:26:53-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码