我的编程空间,编程开发者的网络收藏夹
学习永远不晚

python 泛型函数--singledispatch的使用解读

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

python 泛型函数--singledispatch的使用解读

@functools.singledispatch

将一个函数转变为单一分派的泛型函数

用 @singledispatch装饰一个函数,将定义一个泛型函数。注意,我们创建的函数获得分派的依据是第一个参数的类型:

from functools import singledispatch
@singledispatch
def fun(arg, verbose=False):
    if verbose:
        print("Let me just say,", end=" ")
    print(arg)

使用泛函数的register()属性,重载泛函数的实现。泛函数的register()属性是一个装饰器。对于有类型注释的函数,这个装饰器将自动匹配第一个参为该类型的已注册函数重载泛函数:

@fun.register
def _(arg: int, verbose=False):
    if verbose:
        print("Strength in numbers, eh?", end=" ")
    print(arg)
@fun.register
def _(arg: list, verbose=False):
    if verbose:
        print("Enumerate this:")
    for i, elem in enumerate(arg):
        print(i, elem)

当我们调用泛函数fun时,泛函数根据第一个参数的类型来分派相应的函数来重载实现。

第一个参数为int类型时:

>>> fun(42, verbose=True)
Strength in numbers, eh? 42

第一个参数为list类型时:

>>> fun(['spam', 'spam', 'eggs', 'spam'], verbose=True)
Enumerate this:
0 spam
1 spam
2 eggs
3 spam

如果用泛函数的register()属性进装饰的函数的参数没有类型注释,那么我们可以在register()装饰器中明确声明合适的类型:

@fun.register(complex)
def _(arg, verbose=False):
    if verbose:
        print("Better than complicated.", end=" ")
    print(arg.real, arg.imag)
>>>fun(6+5j, verbose=True)
Better than complicated. 6.0 5.0

为了能注册之前存在的函数和匿名函数,register()属性可以当作功能函数使用。

def nothing(arg, verbose=False):
    print("Nothing.")
    
fun.register(type(None), nothing)
fun.register(int,  lambda x, y, verbose=False: x+y) # 本人添加的,官网没有这个例子

注:经本人实验,如果泛函数出两个可分派的函数,那么,泛涵数将选择离调用最近的可分派的函数,即,泛函数将分派在顺序上最后定义的函数。

>>> fun(None)
Nothing.
>>>fun(1,2)
3

这个register()属性将返回一个未被装饰的函数,这个函数将激活装饰器的堆栈空间,同时为它创建一个独的测试运行单元。

>>>import decimal
>>> @fun.register(float)
... @fun.register(decimal.Decimal)
... def fun_num(arg, verbose=False):
...     if verbose:
...         print("Half of your number:", end=" ")
...     print(arg / 2)
...
>>> fun_num is fun
False

如果泛函数给出的具体类型,没有对应的注册函数的实现,那么泛函数将去寻找更一般化的实现。用@singledispatch装饰的原函数被注册了基本类型–object类型,也就是说如果找不到更好的实现,那么将使用@singledispatch装饰的原函数:

注:此例由本人提供。

>>>fun(bool,verbose=True)
Let me just say, <class 'bool'>

使用只读属性registry,可查看我们都注册了哪些类型的函数实现

>>> fun.registry.keys()
dict_keys([<class 'object'>, <class 'decimal.Decimal'>, <class 'float'>, <class 'int'>, <class 'list'>, <class 'complex'>, <class 'NoneType'>])
>>> fun.registry
{<class 'object'>: <function fun at 0x00000225F21AC268>, <class 'decimal.Decimal'>: <function fun_num at 0x00000225F2517378>, <class 'float'>: <function fun_num at 0x00000225F2517378>, <class 'int'>: <function <lambda> at 0x00000225F2596488>, <class 'list'>: <function _ at 0x00000225F25172F0>, <class 'complex'>: <function _ at 0x00000225F2596400>, <class 'NoneType'>: <function nothing at 0x00000225F258E400>}
>>> fun.registry[float]
<function fun_num at 0x00000225F2517378>
>>>fun.registry[object]
<function fun at 0x00000225F21AC268>

官方链接:https://docs.python.org/3/library/functools.html?highlight=functools wraps#functools.update_wrapper

singledispatch实现单分派泛函数和多分派泛函数

本次的主题是逐渐闯入无人区的泛型!!!

说到泛型,学过java的一定不陌生,泛型的本质是参数化类型,也就是所操作的数据类型被指定为一个参数。但是,学过python的大家是否了解过这部分,或者是使用呢?

那么,python该如何实现泛型呢?

你别说,还真有一个库可以实现!

我们首先导入singledispatch所在的库:

from functools import singledispatch

这个库只能针对函数的第一个参数进行泛型指定!

先指定一个主函数用singledispatch修饰一下,作为一个base, 之后在定义一些“子函数”用 @主函数名.register作为修饰器,并传入一个参数作为“子函数”第一个参数的类型的判断(只能传入一个参数)(这个参数就是“子函数”第一个参数的类型,也是主函数第一个参数的类型)。(注意:这里的子函数就是那个_,应为这个子函数只在泛函数里面会使用到,所以我们干脆不指定他的名字QAQ, 函数的参数也和主函数一样)

但是,这样局限性也太大了,根本没有什么实际用处,我们还要推广到多分派泛函数!!!

多分派泛函数的实现:(因为python只能对第一个参数进行判断泛型,所以我们需要添加一些自己的代码实现多分派反函数)

 我们在单分派的基础上使用isinstance进行了判断,保证其他参数的类型的一致性。

以上的多分派泛函数也可以这样写:

 每一个子函数使用了两个修饰器,但是这两个修饰器都是针对第一个参数的。

!!!你以为这样就完结了???

python 3.5  推出了新特性——参数后面加一个冒号和函数后面加一个->的用法:

(冒号是指该参数应该的参数类型,箭头是指函数应该的返回值)

他也是指定了参数的类型,但是呢,就算你传入的类型和冒号后面的不一样,也并不会报错(除非你有语法错误),所以,这并不是泛型。

但是他和泛型也有些关系,这涉及到了register修饰器的第二个用法:

 省略了register的参数,而使用’:‘符号进行指定。

!!!完结!!!

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

python 泛型函数--singledispatch的使用解读

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java枚举类型与泛型使用解读

这篇文章主要介绍了Java枚举类型与泛型使用解读,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2022-12-28

golang泛型的函数应用

go 1.18 引入了泛型函数,支持类型参数化,增强了代码可重用性。泛型函数语法为 func 函数名[类型参数] (输入参数 类型参数) 类型参数,可使用类型参数化类型作为输入和返回类型。如 min[t number] (a, b t) t
golang泛型的函数应用
2024-04-29

TypeScript 泛型重载函数的使用方式

这篇文章主要介绍了TypeScript 泛型重载函数的使用方式,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
2022-11-13

golang函数类型的广泛应用

go语言中的函数类型允许将函数作为参数或返回值传递,实现灵活的代码。它包括:函数类型声明:func(*argtypes) returntype传递函数作为参数:将函数类型作为其他函数的参数,传递行为。使用函数类型作为返回值:返回函数类型,增
golang函数类型的广泛应用
2024-04-28

如何使用 C++ 函数模板实现函数指针的泛型化?

c++++ 函数模板允许泛化函数指针,支持不同类型参数的函数指针。具体步骤如下:声明带有函数指针的函数模板,其中 t 为模板类型参数。将要泛化的函数指针作为参数传递给模板函数。模板函数返回泛型函数指针。使用 C++ 函数模板实现函数指针的泛
如何使用 C++ 函数模板实现函数指针的泛型化?
2024-04-15

如何创建和使用 C++ 泛型函数指针?

泛型函数指针是一种 c++++ 中指向不同类型和参数数量函数的指针。创建泛型函数指针需要使用模板,其中指定函数的返回值类型和参数类型元组。可通过以下语法使用泛型函数指针:声明函数指针,将函数赋值给函数指针,调用通过函数指针指向的函数。实战案
如何创建和使用 C++ 泛型函数指针?
2024-04-17

C++ 函数库如何使用模版和泛型编程?

c++++ 中的模版和泛型编程允许创建可重用的函数库,其特点包括:模版:参数化类型,在编译时创建类型。泛型编程:利用模版和类型推断编写适用于多种数据类型的代码。实战案例:可重用排序函数可用于任何可比较类型,例如 int 和字符串。C++ 函
C++ 函数库如何使用模版和泛型编程?
2024-04-19

泛型函数在Golang中解决可变参数类型的问题吗?

go 中的泛型函数解决了可变参数类型的问题:泛型函数允许使用类型参数,在运行时指定。这使得编写可以处理不同类型参数的函数成为可能。例如,max 函数是一个泛型函数,它接受两个可比较参数并返回较大值。通过使用泛型函数,我们可以编写更灵活通用的
泛型函数在Golang中解决可变参数类型的问题吗?
2024-04-16

TypeScript中泛型的使用详细讲解

泛型程序设计(genericprogramming)是程序设计语言的一种风格或范式,下面这篇文章主要给大家介绍了关于TypeScript中泛型使用的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2022-11-13

深入了解Rust中泛型的使用

所有的编程语言都致力于将重复的任务简单化,并为此提供各种各样的工具。在 Rust 中,泛型(generics)就是这样一种工具,本文就来聊聊Rust中泛型的使用,需要的可以参考一下
2022-11-13

Java泛型通配符的使用详解

本文主要介绍了Java泛型通配符的使用详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-01-13

C++ 函数模板详解:泛型编程的本质解析

函数模板是 c++++ 中的泛型编程特性,允许创建通用的函数以处理不同类型参数,提高代码灵活性、可重用性和简洁性:定义:函数模板提供函数签名,使用类型参数指定函数操作的数据类型。使用:通过指定类型参数调用函数模板,可用适用于任何类型的通用函
C++ 函数模板详解:泛型编程的本质解析
2024-04-26

如何使用 C++ 函数指针重载和泛型编程?

c++++ 函数指针重载通过指定不同函数签名实现指向具有相同名称但不同参数或返回值的多函数指针。泛型编程使用模板创建适用于不同类型数据的函数和数据结构,使代码可重用。使用函数指针重载需要为每种类型编写单独的函数,而泛型编程则使用通用函数处理
如何使用 C++ 函数指针重载和泛型编程?
2024-04-17

Golang泛型如何解决函数中与类型相关的错误?

泛型解决了 go 函数中与类型相关的错误,允许创建通用函数,可接受并操作各种类型的数据。通过使用类型变量,泛型函数可以适应传递给它们的任何类型,从而避免类型冲突错误,并简化和提高代码的灵活性。Go 泛型用于解决函数中与类型相关的错误在 G
Golang泛型如何解决函数中与类型相关的错误?
2024-04-16

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录