使用python如何实现泛型函数

 更新时间:2022年09月28日 08:25:57   作者:Mr.E5  
这篇文章主要介绍了使用python如何实现泛型函数,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

python实现泛型函数

泛型,即“参数化类型”。一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型。 ——来自百度

简易理解

泛型函数就是你定义函数的时候, 能接收万能类型, 在调用时, 会根据传入值本身的类型进行区分处理, 达到某些效果, 好处是代码复用率高, 减少代码冗余, 对面向对象语言中泛型函数概念非常常用.

接下来使用到的py库 functools 中的 singledispatch 模块

使用方法

在需要进行泛型的函数上加上装饰器即可

from functools import singledispatch
@singledispatch
def add(obj):
    return obj
  • 1 singledispatch : 标记处理函数传值类型
  • 2 register(类型): 为传值判断类型后输出结果
  • 3 后续使用无需写函数名, 只要有register(类型装饰器)即可调用
  • 4 定义需要判断的类型int str tuple dict list set 根据自己需求

函数中实现类型判断

from functools import singledispatch
@singledispatch
def add(obj):
    return obj
int类型
@add.register(int)
def _(add):
    print("int类型")
@add.register(str)
def _(add):
    print("str类型")
    
@add.register(list)
def _(add):
    print("list类型")
    
@add.register(tuple)
def _(add):
    print("tuple类型")
    
@add.register(dict)
def _(add):
    print("dict类型")
    
@add.register(set)
def _(add):
    print("set类型")
    
add([1,2,3])  

输出结果: list类型

根据输入的内容进行判断类型输出

对象中使用

from functools import singledispatch
class Type:
    @singledispatch
    def add(obj):
        return obj
    @add.register(int)
    def _(add):
        print("int类型")
    @add.register(str)
    def _(add):
        print("str")
    @add.register(list)
    def _(add):
        print("list类型")
    @add.register(tuple)
    def _(add):
        print("tuple类型")
    @add.register(dict)
    def _(add):
        print("dict类型")
    @add.register(set)
    def _(add):
        print("set类型")
Type.add([1,2,3])

输出结果:list类型

不调用singledispatch模块实现泛型函数

这里需要实现一个类型拼接操作, 如下代码

在此之前需要先定义一个装饰器, 来判断两个类型是否相同, 如果不同则不作后续判断, 节省资源消耗

def check_type(func):
    def wrapper(*args, **kwargs):
        args1, args2 = args[:2]
        if type(args1) != type(args2):
            return "两种类型不一致, 不能做拼接"
        return func(*args, **kwargs)
    return wrapper
@check_type
def add(obj1, obj2):
    if isinstance(obj1,list):
        obj1 += obj2
        return obj1
    if isinstance(obj1, str):
        obj1 += obj2
        return obj1
    if isinstance(obj1, tuple):
        obj1 += obj2
        return obj1
    if isinstance(obj1, dict):
        obj1 += obj2
        return obj1
print(add([1, 2, 3], [1, 2, 3]))

结果与上方一样, 按需选择

例子很简单, 最后多用于tcp/ip接收判断使用

提示: bool类型也是可以的,完!

Python泛型思考

近日在学习Python内容时学习到了泛型,但从个人看法来说Python泛型与Java的泛型有很大的不同,在此提出一点个人的看法

首先,针对Java的泛型,其主要作用是作为某些以后才指定类型的替代,在编写过程中这些泛型可以在实例化过程中由参数指定,典型例子如集合类当中的泛型

List<String> list = new ArrayList<String>()

这就相当于指定了该集合类的新类型。泛型使用的原因是可以加强类型转化的安全性以及减少转换的次数。比如在上例中的List集合中,如果从中取出一个元素,在未指明泛型的前提下,取出的类型元素只能是Object,必须使用强制类型转换转化为对应的类型才能供后续代码使用。

那么,此处就可能存在问题,因为没有指明泛型,那么意味着加入集合的时候只要你的元素类型是Object即可加入,由于在Java中Object为最高基类,意味着任何元素都可加入泛型当中,由此,没有任何办法保证你取出的元素一定是你想要的的类型,所以此时为保证程序健壮性必须处理ClassCastException异常;但在指明泛型后,由于出入都可以限制元素类型,所以减少了此种转换的异常,也就保证了转换的安全性。

由以上描述可以知道,泛型首先要求的是你能够在实例化时指定类型,所以泛型一般应用于强类型的语言当中才比较好用,但Python本身是弱类型语言,所以Python的泛型并不完全是这种作用

Python的泛型类Generic的注释中提到,

A generic type is typically declared by inheriting from an instantiation of this class with one or more type variables.

也就是说,Python 的泛型类型是继承这个类之后才会声明,这种声明方式与Python的抽象类声明方式十分类似。以IO下的三个实现类为例,其声明分别为

class IO(Generic[AnyStr]):
class BinaryIO(IO[bytes]):
class TextIO(IO[str]):

从这些定义中可以看出,在声明泛型类之后IO类便可以使用泛型的方式去作为其他类的基类定义,

并且AnyStr在Python的类型声明中是包含了bytes与str两类的,由此我们可以发现Python的泛型与Java泛型的一个最根本区别:Python 的泛型是在声明时就指定了泛型类型,也就是说,Python的泛型类型主要作用是:某个类存在多种使用场景,并且我们可以预知各个使用场景的类型,为了将原始代码能够广泛应用于其他场景上使用泛型去保障代码在各个场景上的通用性。因此,在泛型类的声明中并不像其他语言一样使用类似T的方式去替代类型,而是直接使用已经声明的泛型类型在代码中书写,如readline函数的定义

    @abstractmethod
    def readline(self, limit: int = -1) -> AnyStr:
        pass

由此,我们可以归纳Python泛型的两个条件:

1.继承Generic类,并在类的参数中指明泛型类型

2.在实现中使用泛型类型参与代码编写

最后,由于在声明时就已经指定了泛型的类型,所以Python泛型更大程序上我认为是一个具有辅助说明的功能,相当于说明在编写当中告诉编程人员应当使用什么类型,这其实从另一个层面上也是在其他语言中泛型功能的一大体现 

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 图文详解在Anaconda安装Pytorch的详细步骤

    图文详解在Anaconda安装Pytorch的详细步骤

    Anaconda指的是一个开源的Python发行版本,其包含了conda、Python等180多个科学包及其依赖项,下面这篇文章主要给大家介绍了关于在Anaconda安装Pytorch的详细步骤,需要的朋友可以参考下
    2022-07-07
  • Python编程图形库之Pillow使用方法讲解

    Python编程图形库之Pillow使用方法讲解

    今天小编就为大家分享一篇关于Python编程图形库之Pillow使用方法讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • python实现守护进程、守护线程、守护非守护并行

    python实现守护进程、守护线程、守护非守护并行

    本篇文章主要介绍了python实现守护进程、守护线程、守护非守护并行,详细的介绍了守护子进程、非守护子进程并存,守护子线程非守护子进程并存的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • Python 中的函数装饰器和闭包详解

    Python 中的函数装饰器和闭包详解

    这篇文章主要介绍了Python 中的函数装饰器和闭包详解,需要的朋友可以参考下
    2021-02-02
  • Python创建一个元素都为0的列表实例

    Python创建一个元素都为0的列表实例

    今天小编就为大家分享一篇Python创建一个元素都为0的列表实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-11-11
  • python sort、sort_index方法代码实例

    python sort、sort_index方法代码实例

    这篇文章主要介绍了python sort、sort_index方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • Python3批量生成带logo的二维码方法

    Python3批量生成带logo的二维码方法

    今天小编就为大家分享一篇Python3批量生成带logo的二维码方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-06-06
  • python实现跨进程(跨py文件)通信示例

    python实现跨进程(跨py文件)通信示例

    本文主要介绍了python实现跨进程(跨py文件)通信示例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • 30行Python代码实现高分辨率图像导航的方法

    30行Python代码实现高分辨率图像导航的方法

    这篇文章主要介绍了30行Python代码实现高分辨率图像导航的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-05-05
  • 在Python中使用Mako模版库的简单教程

    在Python中使用Mako模版库的简单教程

    这篇文章主要介绍了在Python中使用Mako模版库的简单教程,包括在Django或者Tornado框架中集成Mako的方法,需要的朋友可以参考下
    2015-04-04

最新评论