Python基础篇_修饰符(Decorators)【下】

上一篇:Python基础篇_修饰符(Decorators)【中】@property、@<attribute_name>.setter、@<attribute_name>.deleter、@functools.lru_cache(maxsize=None)

 Python基础篇_修饰符(Decorators)【下】

  • Python基础篇_修饰符(Decorators)【下】
    • 一、修饰符一般特点
    • 二、常用的修饰符以及用法举例
      • 7) @abc.abstractmethod,抽象方法
      • 8) @functools.singledispatch,函数重载
      • 9) @contextlib.contextmanager,上下文管理

Python基础篇_修饰符(Decorators)【下】

Python中有多种修饰符,这些修饰符用于指定方法的特殊行为或属性,也是用于修改函数行为的特殊参数。

一、修饰符一般特点

  1. 修饰符只能用于类定义中,不能用于普通函数中
  2. 属性修饰符是可叠加的,也就是说,一个方法可以同时被多个属性修饰符修饰

二、常用的修饰符以及用法举例

7) @abc.abstractmethod,抽象方法

@abc.abstractmethod用于指示一个方法是抽象的,这意味着这个方法必须在任何非抽象的子类中被重写。它属于abc模块,即抽象基类模块。

当你定义一个抽象基类并使用@abc.abstractmethod装饰器标记一个方法时,任何子类都必须实现这个方法,否则它将引发TypeError

示例1: AbstractClassExample类的抽象方法my_abstract_method

import abcclass AbstractClassExample(metaclass=abc.ABCMeta):@abc.abstractmethoddef my_abstract_method(self):passclass ConcreteClass(AbstractClassExample):def my_abstract_method(self):print("This is the implementation of the abstract method.")# 下面的代码将引发TypeError,因为my_abstract_method在BrokenClass中未被实现。
# class BrokenClass(AbstractClassExample):
#     pass
在这个例子中,AbstractClassExample是一个抽象基类,它有一个抽象方法my_abstract_method
任何继承自AbstractClassExample的子类都必须实现my_abstract_method方法
ConcreteClass是一个实现了该方法的子类
尝试创建一个没有实现该方法的子类(如被注释掉的BrokenClass)将引发TypeError

示例2:Vehicle类的抽象方法startstop

import abcclass Vehicle(metaclass=abc.ABCMeta):@abc.abstractmethoddef start(self):pass@abc.abstractmethoddef stop(self):passclass Car(Vehicle):def start(self):print("Car started!")def stop(self):print("Car stopped!")class Bike(Vehicle):def start(self):print("Bike started!")def stop(self):print("Bike stopped!")# 下面的代码将引发TypeError,因为Vehicle是一个抽象基类,要求子类必须实现start和stop方法。
# class Train(Vehicle):
#     pass
在这个例子中,AbstractClassExample是一个抽象基类,它有一个抽象方法my_abstract_method
任何继承自AbstractClassExample的子类都必须实现my_abstract_method方法
ConcreteClass是一个实现了该方法的子类
尝试创建一个没有实现该方法的子类(如被注释掉的BrokenClass)将引发TypeError

示例3:Shape类的抽象方法area

import abcclass Shape(metaclass=abc.ABCMeta):@abc.abstractmethoddef area(self):passclass Circle(Shape):def __init__(self, radius):self.radius = radiusdef area(self):return 3.14 * self.radius ** 2class Rectangle(Shape):def __init__(self, width, height):self.width = widthself.height = heightdef area(self):return self.width * self.height# 下面的代码将引发TypeError,因为Shape是一个抽象基类,要求子类必须实现area方法。
# class Triangle(Shape):
#     pass    
在这个例子中,Shape是一个抽象基类,有一个抽象方法area。任何子类必须实现这个方法。
Circle和`Rectangle`都是Shape的子类,并实现了area方法。
尝试创建一个没有实现该方法的子类(如被注释掉的Triangle)将引发TypeError

8) @functools.singledispatch,函数重载

@functools.singledispatchfunctools 模块提供的一个装饰器,用于实现多分派。具体来说,它可以将函数重定向到其他函数,基于第一个参数的类型。

示例1:foo方法重载

from functools import singledispatch# 定义一个使用 @singledispatch 的函数
@singledispatch
def foo(arg):raise NotImplementedError("Unsupported type for foo")# 定义一个具体的实现,用于处理字符串类型的参数
@foo.register(str)
def _(arg):return f"You provided a string: {arg}"# 定义一个具体的实现,用于处理整数类型的参数
@foo.register(int)
def _(arg):return f"You provided an integer: {arg}"# 使用函数
print(foo("Hello"))  
print(foo(42))  
print(foo(0.5)) 
# 执行结果
You provided a string: Hello
You provided an integer: 42
NotImplementedError: Unsupported type for foo

示例2:bar方法重载

from functools import singledispatch# 定义一个使用 @singledispatch 的函数
@singledispatch
def bar(arg):raise NotImplementedError("Unsupported type for bar")# 定义一个具体的实现,用于处理字符串类型的参数
@bar.register(str)
def _(arg):return f"You provided a string: {arg}"# 定义一个具体的实现,用于处理整数类型的参数
@bar.register(int)
def _(arg):return f"You provided an integer: {arg}"# 定义一个具体的实现,用于处理列表类型的参数
@bar.register(list)
def _(arg):return f"You provided a list: {arg}"# 使用函数
print(bar("Hello")) 
print(bar(42)) 
print(bar([1, 2, 3])) 
print(bar(0.5)) 
# 执行结果
You provided a string: Hello
You provided an integer: 42
You provided a list: [1, 2, 3]
NotImplementedError: Unsupported type for bar

示例3:baz方法重载

from functools import singledispatch# 定义一个使用 @singledispatch 的函数
@singledispatch
def baz(arg):raise NotImplementedError("Unsupported type for baz")# 定义一个具体的实现,用于处理字符串类型的参数
@baz.register(str)
def _(arg):return f"You provided a string: {arg}"# 定义一个具体的实现,用于处理整数类型的参数
@baz.register(int)
def _(arg):return f"You provided an integer: {arg}"# 定义一个具体的实现,用于处理列表类型的参数
@baz.register(list)
def _(arg):return f"You provided a list: {arg}"# 定义一个更通用的实现,用于处理其他类型
@baz.register
def _(arg):return f"You provided an unknown type: {type(arg)}"# 使用函数
print(baz("Hello"))
print(baz(42))
print(baz([1, 2, 3]))
print(baz(0.5))
# 执行结果
You provided a string: Hello
You provided an integer: 42
You provided a list: [1, 2, 3]
You provided an unknown type: <class 'float'>

9) @contextlib.contextmanager,上下文管理

@contextlib.contextmanager 用于简化上下文管理器的创建。上下文管理器允许你在代码的某个特定部分设置一个上下文,该上下文在其他部分可能无法访问或可能更改。常见的使用场景包括文件操作、线程锁等。

使用 @contextlib.contextmanager 装饰器可以使您以声明式方式编写上下文管理器,而无需实现 __enter____exit__ 方法。这对于简化某些上下文管理任务非常有用。

示例1:名为 timer 的上下文管理器

接受一个名称参数并打印出开始和结束时间

import contextlib
import time@contextlib.contextmanager
def timer(name):print(f"Starting {name}")start_time = time.time()try:yieldfinally:end_time = time.time()print(f"{name} took {end_time - start_time} seconds")with timer("my_operation"):# 在这里执行需要计时的操作time.sleep(2)
# 执行结果
Starting my_operation
my_operation took 2.00 seconds

示例2:名为 redirect_stdout 的上下文管理器

将标准输出(通常显示在控制台)重定向到一个指定的文件

import contextlib
import os@contextlib.contextmanager
def redirect_stdout(file_path):"""将标准输出重定向到指定的文件"""original_stdout = sys.stdouttry:with open(file_path, 'w') as f:sys.stdout = fyieldfinally:sys.stdout = original_stdout# 使用重定向输出的上下文管理器
with redirect_stdout('output.txt'):print("This message will be written to the file.")

示例3:名为 thread_lock 的上下文管理器

接受一个 threading.Lock 对象作为参数。这个上下文管理器确保在 with 语句块中的代码在执行时被线程锁定

import contextlib
import threading@contextlib.contextmanager
def thread_lock(lock):"""使用线程锁的上下文管理器:param lock: threading.Lock 对象"""lock.acquire()try:yieldfinally:lock.release()# 创建一个线程锁对象
lock = threading.Lock()# 使用线程锁的上下文管理器
with thread_lock(lock):# 在此块中的代码将被线程锁定,确保同一时间只有一个线程可以执行这段代码print("Doing critical section of code...")

may the odds be ever in your favor ~

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

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

相关文章

ElasticSearch之search API

写在前面 本文看下查询相关内容&#xff0c;这也是我们在实际工作中接触的最多的&#xff0c;所以有必要好好学习下&#xff01; 1&#xff1a;查询的分类 主要分为如下2类&#xff1a; 1:基于get查询参数的URI search 2&#xff1a;基于post body的request body search&am…

【算法】基础算法模板

文章目录 一、快速排序二、归并排序三、二分1. 二分的本质2. 整数二分3. 实数二分 四、前缀和1. 一维前缀和2. 二维前缀和 五、差分1. 一维差分2. 二维差分 六、常用位运算1. 求二进制的第 k 位2. lowbit 七、其他常用算法1. 去重2. 表达式求值3. 单调栈4. 单调队列5. 并查集 一…

Go语言安全编码:crypto/sha1库全面解析

Go语言安全编码&#xff1a;crypto/sha1库全面解析 简介SHA-1基础原理和特点SHA-1与其他哈希算法的比较代码示例&#xff1a;基本的SHA-1哈希生成 使用crypto/sha1处理数据处理字符串和文件的SHA-1哈希代码示例&#xff1a;为文件生成SHA-1哈希 常见错误和最佳实践 在实际项目中…

leetcode(双指针)283.移动零(C++详细题解)DAY3

文章目录 1.题目示例提示 2.解答思路3.实现代码结果 4.总结 1.题目 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 请注意 &#xff0c;必须在不复制数组的情况下原地对数组进行操作。 示例 示例 1: 输入…

fpga 需要掌握哪些基础知识?

个人根据自己的一些心得总结一下fpga 需要掌握的基础知识&#xff0c;希望对你有帮助。 1、数电&#xff08;必须掌握的基础&#xff09;&#xff0c;然后进阶学模电&#xff0c; 2、掌握HDL&#xff08;verilog或VHDL&#xff09;一般建议先学verilog&#xff0c;然后可以学…

【笔记】Helm-5 Chart模板指南-9 在模板内部访问文件

在模板内部访问文件 在上一节中&#xff0c;我们研究了几种创建和访问模板的方法。这样可以很容易从一个模板导入到另一个模板中。但有时想导入的是不是模板的文件并注入其内容&#xff0c;而无需通过模板渲染发送内容。 Helm提供了通过.Files对象访问文件的方法。不过&#x…

【Make编译控制 06】CMake初步使用

目录 一、概述与安装 二、编译源文件 三、无关文件管理 一、概述与安装 CMake是一个跨平台的项目构建工具&#xff0c;相比于Makefile&#xff0c;CMake更加高级&#xff0c;因为CMake代码在执行的时候是会先翻译生成Makefile文件&#xff0c;再调用Makefile文件完成项目构…

辅警考试怎么搜答案?这4款足够解决问题 #微信#知识分享

对于大学生来说&#xff0c;每天面对各式各样的学习任务和问题&#xff0c;寻找合适的学习资源和工具成了我们的迫切需求。幸运的是&#xff0c;现如今有许多高效且实用的日常搜题和学习软件可以满足我们的需求&#xff0c;助力我们取得更好的学习成果。 1.颐博查题 这是一个…

Django问题报错:TypeError: as_view() takes 1 positional argument but 2 were given

一、错误位置 from django.urls import pathfrom users_app.views import RegisterView, LoginView, LogoutViewapp_name users urlpatterns [path("register/", RegisterView.as_view, name"register"),path("login/", LoginView.as_view, n…

Netty中的常用组件(三)

ChannelPipeline 基于Netty的网路应用程序中根据业务需求会使用Netty已经提供的Channelhandler 或者自行开发ChannelHandler&#xff0c;这些ChannelHandler都放在ChannelPipeline中统一 管理&#xff0c;事件就会在ChannelPipeline中流动&#xff0c;并被其中一个或者多个Chan…

VED-eBPF:一款基于eBPF的内核利用和Rootkit检测工具

关于VED-eBPF VED-eBPF是一款功能强大的内核漏洞利用和Rootkit检测工具&#xff0c;该工具基于eBPF技术实现其功能&#xff0c;可以实现Linux操作系统运行时内核安全监控和漏洞利用检测。 eBPF是一个内核内虚拟机&#xff0c;它允许我们直接在内核中执行代码&#xff0c;而无…

什么是Java中的NIO(New I/O)?与传统的I/O有什么不同?

什么是Java中的NIO&#xff08;New I/O&#xff09;&#xff1f;与传统的I/O有什么不同&#xff1f; Java NIO&#xff08;New I/O&#xff09;是Java 1.4引入的一组用于非阻塞I/O操作的API。与传统的I/O&#xff08;即普通的阻塞I/O&#xff09;相比&#xff0c;Java NIO提供…

CVE-2022-0760 漏洞复现

CVE-2022-0760 NSS [HNCTF 2022 WEEK2]ohmywordpress 【CVE-2022-0760】 题目描述&#xff1a;flag在数据库里面。 开题&#xff1a; 顺着按钮一直点下去会发现出现一个按钮叫安装WordPress 安装完之后的界面&#xff0c;有一个搜索框。 F12看看network。 又出现了这个Wor…

华为数通方向HCIP-DataCom H12-821题库(单选题:461-480)

第461题 以下关于路由策略特点的描述,错误的是哪一项? A、能够修改路由属性,但是不能改变网络流量经过的路径 B、能通过控制路由器的路由表规模,来节约系统资源 C、能通过控制路由的接收、发布和引入,以提高网络的安全性 D、能通过修改路由属性,对网络数据流量可以合理规…

Python爬虫之关系型数据库存储#5

关系型数据库是基于关系模型的数据库&#xff0c;而关系模型是通过二维表来保存的&#xff0c;所以它的存储方式就是行列组成的表&#xff0c;每一列是一个字段&#xff0c;每一行是一条记录。表可以看作某个实体的集合&#xff0c;而实体之间存在联系&#xff0c;这就需要表与…

Java异常的处理 try-catch-finally

目录 什么是异常通过if-else处理异常用if-else堵漏洞的缺点 try-catch例第一种处理第二种处理第三种处理第四种处理 try-catch-finally例 System.exit(0);//终止当前的虚拟机执行 什么是异常 Exception&#xff1a;在程序的运行过程中&#xff0c;发生了不正常的现象&#xff0…

SpringCloud-Ribbon实现负载均衡

在微服务架构中&#xff0c;负载均衡是一项关键的技术&#xff0c;它可以确保各个服务节点间的负载分布均匀&#xff0c;提高整个系统的稳定性和性能。Spring Cloud 中的 Ribbon 就是一种负载均衡的解决方案&#xff0c;本文将深入探讨 Ribbon 的原理和在微服务中的应用。 一、…

12 ABC串口接收原理与思路

1. 串口接收原理 基本原理&#xff1a;通过数据起始位判断要是否要开始接收的数据&#xff0c;通过采样的方式确定每一位数据是0还是1。 如何判断数据起始位到来&#xff1a;通过边沿检测电路检测起始信号的下降沿 如何采样&#xff1a;一位数据采多次&#xff0c;统计得到高…

curl8.6.0 - CURLE_PEER_FAILED_VERIFICATION

文章目录 curl8.6.0 - CURLE_PEER_FAILED_VERIFICATION概述笔记END curl8.6.0 - CURLE_PEER_FAILED_VERIFICATION 概述 在看一个开源工程, 里面用到了curl和openssl, 但是工程使用vcpkg来管理的包, 用CMake来编译 依赖太多了, win10 编译选项为 vs2019 x64/Win32(或者Ninja)…

从零开始:用 Rust 编写你的第一个 Web 服务

Rust 是一种现代、高性能的编程语言&#xff0c;近年来在 Web 开发领域也有了一席之地。本文将介绍如何使用 Rust 编写一个简单的 Web 程序&#xff0c;从搭建开发环境到创建第一个 Web 页面。 1. 开发环境搭建 首先&#xff0c;确保你已经安装了 Rust 工具链。你可以通过在终…