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.xdnf.cn/news/34890.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

12.4深度学习_模型优化和迁移_awanb、tb

一、数据获取方法 1. 开源数据集 ​ 免费,成本低 PyTorch: https://pytorch.org/vision/stable/datasets.html 开源数据集imagenet:https://image-net.org/ Hugging Face数据集:https://huggingface.co/datasets kaggle数据集…

网络基础知识

172.16.24.100这个是ip地址,讲师机的IP地址。IP地址(Internet Protocol Address)是指互联网协议地址,又译为网际协议地址。每台电脑只要联网都会有ip地址。ip地址数量有限,不够给世界上每一台电脑分配ip地址&#xff0…

漫画之家系统:Spring Boot技术下的漫画发现引擎

4 系统设计 4.1系统设计主要功能 通过市场调研及咨询研究,了解了用户及管理者的使用需求,于是制定了管理员和用户等模块。功能结构图如下所示: 图4-1系统功能结构图 4.2数据库设计 4.2.1数据库设计规范 数据可设计要遵循职责分离原则&#…

漫画之家系统:Spring Boot框架下的漫画版权保护

摘 要 随着信息技术和网络技术的飞速发展,人类已进入全新信息化时代,传统管理技术已无法高效,便捷地管理信息。为了迎合时代需求,优化管理效率,各种各样的管理系统应运而生,各行各业相继进入信息管理时代&a…

【python rich 超级牛终端中提供富文本和精美格式】

Rich 是一个 Python 库,可以为您在终端中提供富文本和精美格式。 》》》》官方代码和文档《《《《 Rich 的 API 让在终端输出颜色和样式变得很简单。此外,Rich 还可以绘制漂亮的表格、进度条、markdown、语法高亮的源代码以及栈回溯信息(tr…

【电子设计】WifiESP8266无线通信

硬件 野火STM32开发板 操作系统 FreeRTOS 软件Keil5野火蓝牙模块 ESP8266模块 1. ESP8266 简介 ESP8266 是串口型 WIFI,速度比较低,不能用来传输图像或者视频这些大容量的数据,主要应用于数据量传输比较少的场合,比如温湿度…

44.5.【C语言】辨析“数组指针”和“指针数组”

目录 1.数组指针 2.指针数组 执行结果 底层分析 1.数组指针 从语文的角度理解,"数组"修饰"指针".因此数组指针是指针 例如以下代码 #include <stdio.h> int main() {char a[5] { "ABCDE" };return 0;} 其中a就是数组指针,因为数…

docker安装victoriametrics(单机版)

docker安装victoriametrics 1、单机版安装2、victoriametrics增删改查2.1 、插入数据2.1.1 组装数据插入victoriametrics(java代码插入)2.1.2 Prometheus数据插入victoriametrics2.1.3 官网push到victoriametrics写法 2.2 、查询2.2.1 、Instant query&#xff08;即时查询&…

趣讲TCP三次握手

一、TCP三次握手简介 TCP&#xff08;Transmission Control Protocol&#xff0c;传输控制协议&#xff09;是一种面向连接的、可靠的、基于字节流的传输层通信协议。在TCP连接中&#xff0c;只有两方进行通信&#xff0c;它使用校验和、确认和重传机制来保证数据的可靠传输。…

攻防世界 ctf刷题 新手区1-10

unserialize3 因为我上个笔记写了 php返序列化 所以先趁热打铁 看这个题目名字 我们就知道是 反序列化呀 因为flag有值所以 我们先输个 111 看看有没有线索 没线索但是这边 有个发现就是他是使用get方式传参的 可能他会把我们的输入 进行传入后台有可能进行反…

股指期货基差的影响因素有哪些?

在股指期货交易中&#xff0c;有一个重要的概念叫做“基差”。简单来说&#xff0c;基差就是股指期货价格与其对应的现货价格之间的差异。比如&#xff0c;我们现在有IC2401股指期货&#xff0c;它挂钩的是中证500指数。如果IC2401的价格是5244&#xff0c;而中证500指数的价格…

【单片机基础知识】MCU三种启动方式(Boot选择)[主Flash/系统存储器(BootLoader)/嵌入式SRAM]——老版

请跳转到最新版&#xff1a; 【单片机开发】MCU三种启动方式(Boot选择)[主Flash/系统存储器(BootLoader)/嵌入式SRAM]-CSDN博客 参考资料&#xff1a; MCU的三种启动方式 - EdgeAI Lab 立芯嵌入式的视频 在SRAM中运行代码 - EdgeAI Lab 利用 Boot 选择不同的启动方式&…

frp内网穿透的配置与设置

FRP&#xff08;Fast Reverse Proxy&#xff09;是一个高性能的反向代理应用&#xff0c;可以实现内网穿透功能。它帮助你将内网的服务暴露到公网&#xff0c;无需公网IP和端口映射&#xff0c;非常适合需要穿透防火墙、NAT的场景。以下是 FRP 内网穿透的配置和设置方法。 ###…

图数据库 | 13、图数据库架构设计——高性能计算架构再续

书接上文 图数据库 | 12、图数据库架构设计——高性能计算架构​​​​​​。昨天老夫就图数据库架构设计中的 实时图计算系统架构、图数据库模式与数据模型、核心引擎如何处理不同的数据类型、图计算引擎中的数据结构 这四块内容进行了展开讲解&#xff0c;今儿继续往下、往深…

一、web基础和http协议

前言 https://www.baidu.com/&#xff1a;URL&#xff08;是一种万维网寻址网址&#xff09; https://&#xff1a;协议&#xff0c;加密的http&#xff0c;加密的超文本传输协议&#xff0c;在数据传输之前要通过整数进行身份验证&#xff0c;验证通过才可以进行数据传输。 …

基于java+SpringBoot+Vue的实验室管理系统设计与实现

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; Springboot mybatis Maven mysql5.7或8.0等等组成&#x…

win7 双机调试

32位 CMD&#xff1a;关闭指令(开启10-10-12分页) bcdedit /set pae ForceDisable bcdedit /set nx AlwaysOff 开启指令(开启2-9-9-12分页) bcdedit /set pae forceEnable bcdedit /set nx OptIn一件事情是配置好虚拟机&#xff0c;我这里使用…

Linux进程优先级

1.孤儿进程 父子进程关系中&#xff0c;如果父进程先退出&#xff0c;子进程要被1号进程领养&#xff0c;这个被领养的进程为孤儿进程(1号进程为操作系统)&#xff0c;不领养如果变成僵尸进程就会有弊端&#xff0c;只有父进程和操作系统可以回收子进程&#xff0c;孤儿进程是…

力扣1401. 圆和矩形是否有重叠

用矢量计算&#xff1a; class Solution { public:bool checkOverlap(int radius, int xCenter, int yCenter, int x1, int y1, int x2, int y2) {//矩形中心float Tx(float)(x1x2)/2;float Ty(float)(y1y2)/2;//强行进行对称操作&#xff0c;只考虑第一象限if(xCenter<Tx)…

GESP202303 一级【长方形面积】题解(AC)

》》》点我查看「视频」详解》》》 AC_Code #include <bits/stdc.h> using namespace std;int main() {int n;cin >> n;int cnt 0;for(int i 1; i < n; i )for(int j i; j < n; j )if(i * j n)cnt ;cout << cnt;return 0; }》》》点我查看「视频」…