python 装饰器学习与实践

目录

  • 装饰器学习
    • 1、最基本装饰器
    • 2、函数带参数的装饰器
    • 3、装饰器带参数
    • 4、类中函数的装饰器
    • 5、装饰器实践
    • 6、pyqt5类中方法的装饰器实现时遇到的问题

装饰器学习

先假定一个场景
在之前的一篇文章中,分享了一个pyqt5将日志实时展示在gui界面上的功能python在pyqt5+logging+threading模块实时显示日志,现在我们需要当函数发生异常时能将异常信息也打印在界面上,以方便我们对程序进行优化,实现结果类似如下:
在这里插入图片描述
但我们程序已经写好,如果对每个函数内容都去做try-except处理无疑是很费时的,这里我们就可以用装饰器对函数进行包装

1、最基本装饰器

装饰器,对现有函数进行装饰,可以在函数执行前、后编写程序来增加功能,实际就是在函数外层嵌套一层函数

# 接收一个函数(不带参数),并在函数执行后,打印一句话
def outer(func):def inner():func()print('我是一个装饰器')return inner@outer
def a():print('我是一个普通函数')if __name__=='__main__':a()print(a.__name__)

执行结果如下:
在这里插入图片描述
从图中我们可以看到,第1 ,函数执行后,正常执行了装饰器的语句;第2,函数名变了,这是因为使用一般装饰器时,实际的执行过程是:outer(a)->inner(),inner才是他最终执行的函数,这里如果不想函数名称被装饰器影响,可以使用functools.wraps方法包装传入的函数名:

def outer(func):@functools.wraps(func)def inner():func()print('我是一个装饰器')return inner

包装后执行结果如下:
在这里插入图片描述

2、函数带参数的装饰器

在实际编程过程中,我们的函数实际上会带上各种参数的,要想装饰器能适用于各种参数的函数,我们需要在装饰器中接收函数传递进来的参数,如果被装饰的函数入参格式各不相同,我们可以使用如下写法:

# 函数带参数的装饰器
def outer1(func):@functools.wraps(func)def inner(*args, **kwargs):		# 在inner函数后加入参数,即可适用于装饰带参数的函数print('====我是装饰器,装饰函数%s===='%func.__name__)print('位置参数是:',args)print('关键字参数是:',kwargs)func(*args, **kwargs)return inner# 不带参数的函数
@outer1
def test1():print('我是一个普通函数')# 带位置参数的函数
@outer1
def test2(a):print('我带一个位置参数,值是:', a)# 带位置参数和关键字参数的函数
@outer1
def test3(a, b, c='关键1', d='关键2'):print('我带位置参数和关键字参数,他们的值是:', a, b, c, d)if __name__ == '__main__':test1()test2('测试')test3('测试1', '测试2', c='测试3')

执行结果如下:
在这里插入图片描述

3、装饰器带参数

假设一个场景,在对多个函数进行装饰的同时又希望加入额外的不同处理,比如对于上一小节的test1、test2、test3,假设他们都需要一个传入一个系数和一个名称才能进行后续计算。我们可以在现有装饰器基础上再往外嵌套一层,来实现自定义参数带入,如下:

# 装饰器带参数
def outerouter(k, name='test'):def outer1(func):@functools.wraps(func)def inner(*args, **kwargs):print('====我是装饰器,装饰函数%s====' % func.__name__)print('我的名称和系数分别是', name, k)print('位置参数是:', args)print('关键字参数是:', kwargs)func(*args, **kwargs)return innerreturn outer1@outerouter(1, '测试1')
def test1():print('我是一个普通函数')# 带位置参数的函数
@outerouter(2.22, '测试2')
def test2(a):print('我带一个位置参数,值是:', a)# 带位置参数和关键字参数的函数
@outerouter(3.3333, '测试3')
def test3(a, b, c='关键1', d='关键2'):print('我带位置参数和关键字参数,他们的值是:', a, b, c, d)if __name__ == '__main__':test1()test2('测试')test3('测试1', '测试2', c='测试3')

执行结果如下:
在这里插入图片描述

4、类中函数的装饰器

写一个类,我们把之前章节的test3函数放到类中,如下:

class A:@outer_class(1.234, '类')def test(self, a, b, c='关键1', d='关键2'):print('我带位置参数和关键字参数,他们的值是:', a, b, c, d)if __name__ == '__main__':a = A()a.test3('测试1', '测试2', c='测试3')

由于类中函数实际执行时,第一个位置参数始终都是实例本身,如上,我们在调用实例方法a.test3(‘测试1’, ‘测试2’, c=‘测试3’)时,实际上是执行的test3(a,‘测试1’, ‘测试2’, c=‘测试3’),因此对类中的实例方法,他的第一个参数始终是实例本身,我们将第3节的装饰器做如下修改后,即可对其进行装饰:

# 类中函数的装饰器
def outer_class(k, name='test'):def outer1(func):@functools.wraps(func)def inner(self, *args, **kwargs):   # 第一个位置参数是固定,代指实例print('====我是装饰器,装饰函数%s====' % func.__name__)print('我的名称和系数分别是', name, k)print('位置参数是:', args)print('关键字参数是:', kwargs)func(self, *args, **kwargs)     # 第一个位置参数是固定,代指实例return innerreturn outer1class A:@outer_class(1.234, '类')def test3(self, a, b, c='关键1', d='关键2'):print('我带位置参数和关键字参数,他们的值是:', a, b, c, d)if __name__ == '__main__':a = A()a.test3('测试1', '测试2', c='测试3')

执行结果如下:
在这里插入图片描述

5、装饰器实践

了解装饰器后,我们对之前的日志显示工具做优化: python在pyqt5+logging+threading模块实时显示日志

  1. 优化之前的点击按钮后函数,移除方法中的线程创建代码,让方法只专注于解决专一问题,并给一个假的异常抛出用以试验日志装饰器:
    def log_print(self):"""按钮触发的函数"""for i in range(3):logger.info('我正在打印日志%s' % i)logger.info('我等待一秒')time.sleep(1)raise Exception('执行结束了,抛出异常!')
  1. 写一个日志装饰器,用于将异常打印到gui界面
def log_info(func):@functools.wraps(func)def inner(self):try:func(self)except:logger.info(traceback.format_exc())return inner
  1. 写一个创建线程的装饰器,用以给指定方法创建线程,合方法打印能实时显示在gui界面上
def thread_func(func):@functools.wraps(func)def inner(self):t = threading.Thread(target=func,args=(self,))t.start()return inner
  1. 先装饰日志,再装饰多线程(注意一定要先装饰日志函数,否则日志不会打印到界面上,因为线程中的异常不会返回到主线程中,所以这里直接把日志装饰在线程函数中实现),执行结果如下:
    在这里插入图片描述
    两个装饰器分边实现日志打印到界面与多线程实现日志实时打印

6、pyqt5类中方法的装饰器实现时遇到的问题

在pyqt5中的信号触发方法进行装饰器的时候,例如日志装饰器,如果想适用于所有带各种参数的函数,则需要在装饰器中使用*args和**kwargs,就如我们在第4节中所讲:

def log_info(func):@functools.wraps(func)def inner(self, *args, **kwargs):try:func(self, *args, **kwargs)except:logger.info(traceback.format_exc())return innerdef thread_func(func):@functools.wraps(func)def inner(self, *args, **kwargs):t = threading.Thread(target=func, args=(self, *args,), kwargs=kwargs)t.start()return inner

但实际在pyqt5的类中执行时发生异常
在这里插入图片描述
打断点调试后发现,入参时args多传了一个False
在这里插入图片描述
目前无法确定是哪一步传的这个False,有可能是pyqt5的信号触发的这个,log_bt.clicked.connect(self.log_print),如果有老师知道的话感谢评论解答一下,感谢!
在这里插入图片描述
目前的暂时的处理办法是取args时取后面切片[1:]在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/485011.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OCR的评价指标和常用数据集

1.OCR任务简介 OCR(Optical Character Recognition,光学字符识别)是指对包含文本内容的图像或者视频进行处理识别,并提取其中所包含的文字及排版信息的过程。例如,一个常见的应用是将包含文档图像的不可编辑状态的 PDF 文档通过 OCR 技术识别后&#xf…

解决el-select数据量过大的3种方法

在准备上线的后台管理系统中,我们发现有两个下拉框(select),其选项数据量超过 1 万条,而在测试环境中这些数据量只有几百条。这导致在页面加载时,浏览器性能出现瓶颈,页面卡顿甚至崩溃。 想了一…

计算机网络 第5章 运输层

计算机网络 (第8版) 第 5 章 传输层5.4 可靠传输的原理5.4.1 停止等待协议5.4.2 连续ARQ协议 5.5 TCP报文段的首部格式5.6 TCP可靠传输的实现5.6.1 以字节为单位的滑动窗口5.6.2 超时重传时间的选择 5.7 TCP的流量控制5.7.1 利用滑动窗口实现流量控制 5.…

【PyTorch】(基础三)---- 图像读取和展示

图像读取和展示 pytorch本身并不提供图像的读取和展示功能,利用pytorch执行计算机视觉任务的时候,通常是利用opencv等工具先进行图像处理,然后将结果转化成tensor类型传递给pytorch,在pytorch执行之后,也可以将tensor…

Java课程设计项目-servlet+jsp美食系统、菜品管理系统

文章目录 Java课程设计项目-servletjsp美食系统一、项目介绍二、技术介绍2.1 环境需要2.2 技术栈 环境需要三、功能实现3.1登录注册3.2首页菜品展示、轮播图3.3美食菜品分类、查询3.4作品动态、个人简介、菜品收藏3.5创建菜谱、添加步骤 四、系统代码展示4.1项目架构&#xff0…

使用Unity脚本模拟绳索、布料(碰撞)

效果演示: 脚本如下: using System.Collections; using System.Collections.Generic; using UnityEngine;namespace PhysicsLab {public class RopeSolver : MonoBehaviour {public Transform ParticlePrefab;public int Count 3;public int Space 1;…

Python 【图像分类】之 PyTorch 进行猫狗分类功能的实现(Swanlab训练可视化/ Gradio 实现猫狗分类 Demo)

Python 【图像分类】之 PyTorch 进行猫狗分类功能的实现(Swanlab训练可视化/ Gradio 实现猫狗分类 Demo) 目录 Python 【图像分类】之 PyTorch 进行猫狗分类功能的实现(Swanlab训练可视化/ Gradio 实现猫狗分类 Demo) 一、简单介绍 二、PyTorch 三、CNN 1、神经网络 2、卷…

【Python网络爬虫笔记】8- (BeautifulSoup)抓取电影天堂2024年最新电影,并保存所有电影名称和链接

目录 一. BeautifulSoup的作用二. 核心方法介绍2.1 构造函数2.2 find()方法2.3 find_all()方法2.4 select()方法 三. 网络爬虫中使用BeautifulSoup四、案例爬取结果 一. BeautifulSoup的作用 解析HTML/XML文档:它可以将复杂的HTML或XML文本转换为易于操作的树形结构…

ZLMediaKit+wvp (ffmpeg+obs)推拉流测试

这里使用了两种方式: ffmpeg命令和 OBS OBS推流在网上找了些基本没有说明白的, 在ZLMediaKit的issues中看到了一个好大哥的提问在此记录一下 使用OBS推流,rtmp,报鉴权失败 推流 1. ffmpeg命令推流 官方说明文档地址: 推流规则 rtsp://192.168.1.4:10554…

Linux入门攻坚——40、Linux集群系统入门-lvs(1)

Cluster,集群,为了解决某个特定问题将多台计算机组合起来形成的单个系统。 这个单个集群系统可以扩展,系统扩展的方式:scale up,向上扩展,更换更好的主机;scale out,向外扩展&…

威胁驱动的网络安全方法论

本文主要内容取自洛克希德马丁公司的论文:A Threat-Driven Approach to Cyber Security,想要全面准确了解论文内容的朋友建议阅读原文。希望能够抛砖引玉,为相关领域的相关工作人员带来一点不同的思路或启发,从而更好地维护企业/组…

【Verilog】实验三 数码管实验

目录 一、实验目的: 二、实验内容: 三、实验要求: 四、实验步骤: 一、实验目的: 进一步熟悉Modelsim和VIVADO工具;掌握7段数码管显示译码器;掌握7段数码管数码管动态输出显示的方法。 二、实验内容: 实现按动开关…

Spring Cloud + MyBatis Plus + GraphQL 完整示例

Spring Cloud MyBatis Plus GraphQL 完整示例 1、创建Spring Boot子项目1.1 配置POM,添加必要的依赖1.2 配置MyBatis-Plus 2、集成GraphQL2.1 定义schema.graphqls2.2 添加GraphQL解析器2.3 配置schame文件配置 3、访问测试3.1 查询测试(演示&#xff…

MySQL书籍推荐

《高性能MySQL(第4版)》-西尔维亚博特罗斯 系统层次 Mysql性能优化和高可用架构实践 2020 系统基础 MySQL性能调优与架构设计 系统基础 Mysql技术大全 2021 综合 MySQL数据库应用案例教程 综合实战 从入门到项目实践 综合实战 丰富 超值 MySQ…

MR30分布式IO模块赋能喷水织机

纺织行业作为我国传统支柱产业,历经数千年的演变,如今仍面临着诸多困境,在纺织行业中,每一次技术的飞跃都是对行业边界的勇敢探索。在纺织行业,喷水织机作为关键生产设备,其性能直接影响到产品质量和产能。…

nodejs循环导出多个word表格文档

文章目录 nodejs循环导出多个word表格文档一、文档模板编辑二、安装依赖三、创建导出工具类exportWord.js四、调用五、效果图nodejs循环导出多个word表格文档 结果案例: 一、文档模板编辑 二、安装依赖 // 实现word下载的主要依赖 npm install docxtemplater pizzip --save/…

LabVIEW中“this VI‘s owning library is missing”错误及解决

问题描述 当加载或打开一个VI时,如果其所属的项目库未加载到内存,LabVIEW将提示错误:“this VIs owning library is missing”(该VI的所属库不存在)。 该问题通常发生在以下情况下: 项目库文件丢失或路径…

LongVU:用于长视频语言理解的空间时间自适应压缩

晚上闲暇时间看到一种用于长视频语言理解的空间时间自适应压缩机制的研究工作LongVU,主要内容包括: 背景与挑战:多模态大语言模型(MLLMs)在视频理解和分析方面取得了进展,但处理长视频仍受限于LLM的上下文长…

sphinx基本使用

sphix是一个文档生成工具 本文介绍一些基础技能,如果想深入学习,可以查看官方文档 Sphinx官方文档 1.安装虚拟环境 # ubuntu # 使用 venv 创建 .venv虚拟环境 python3 -m venv .venv# 激活虚拟环境 source .venv/bin/activate# windows # 创建虚拟环境…

爬虫第四篇:Xpath 路径表达式全解析:从网页基础到爬取百度贴吧图片实战

简介:本文围绕 Xpath 路径表达式展开讲解,先是介绍了网页相关基础如 html、css、vue 以及前后端分离的概念与示例,包括各部分的结构、作用及简单代码展示,随后详细阐述了 xml 的节点关系、选取节点、谓语等理论知识,最…