泛型,如果你嘗過java,應該對他不陌生吧。但你可能不知道在 Python 中(3.4+ ),也可以實現 簡單的泛型函數。在Python中只能實現基于單個(第一個)參數的數據類型來選擇具體的實現方式,官方名稱 是single-dispatch。你或許聽不懂,說人話,就是可以實現第一個參數的數據類型不同,其調用的函數也就不同。它使用方法極其簡單,只要被singledispatch裝飾的函數,就是一個single-dispatch的泛函數(generic functions)。單分派:根據一個參數的類型,以不同方式執行相同的操作的行為。
多分派:可根據多個參數的類型選擇專門的函數的行為。
泛函數:多個函數綁在一起組合成一個泛函數。
這邊舉個簡單的例子。from functools import singledispatch@singledispatchdef age(obj):print('請傳入合法類型的參數!')@age.register(int)def _(age):print('我已經{}歲了。'.format(age))@age.register(str)def _(age):print('I am {} years old.'.format(age))age(23) # intage('twenty three') # strage # list
執行結果我已經23歲了。I am twenty three years old.請傳入合法類型的參數!說起泛型,其實在 Python 本身的一些內建函數中并不少見,比如,pprint等
你可能會問,它有什么用呢?實際上真沒什么用,你不用它或者不認識它也完全不影響你編碼。
我這里舉個例子,你可以感受一下。
大家都知道,Python 中有許許多的數據類型,比如 str,list, dict, tuple 等,不同數據類型的拼接方式各不相同,所以我這里我寫了一個通用的函數,可以根據對應的數據類型對選擇對應的拼接方式拼接,而且不同數據類型我還應該提示無法拼接。以下是簡單的實現。def check_type(func):def wrapper(*args):arg1, arg2 = argsif type(arg1) != type(arg2):return '【錯誤】:參數類型不同,無法拼接!!'return func(*args)return wrapper@singledispatchdef add(obj, new_obj):raise TypeError@add.register(str)@check_typedef _(obj, new_obj):obj += new_objreturn obj@add.register(list)@check_typedef _(obj, new_obj):obj.extend(new_obj)return obj@add.register(dict)@check_typedef _(obj, new_obj):obj.update(new_obj)return obj@add.register(tuple)@check_typedef _(obj, new_obj):return (*obj, *new_obj)print(add('hello',', world'))print(add)print(add({'name': 'wangbm'}, {'age':25}))print(add(('apple', 'huawei'), ('vivo', 'oppo')))# list 和 字符串 無法拼接print(add(, '4,5,6'))
輸出結果如下hello, world{'name': 'wangbm', 'age': 25}('apple', 'huawei', 'vivo', 'oppo')【錯誤】:參數類型不同,無法拼接!!
如果不使用singledispatch 的話,你可能會寫出這樣的代碼。def check_type(func):def wrapper(*args):arg1, arg2 = argsif type(arg1) != type(arg2):return '【錯誤】:參數類型不同,無法拼接!!'return func(*args)return wrapper@check_typedef add(obj, new_obj):if isinstance(obj, str) :obj += new_objreturn objif isinstance(obj, list) :obj.extend(new_obj)return objif isinstance(obj, dict) :obj.update(new_obj)return objif isinstance(obj, tuple) :return (*obj, *new_obj)print(add('hello',', world'))print(add)print(add({'name': 'wangbm'}, {'age':25}))print(add(('apple', 'huawei'), ('vivo', 'oppo')))# list 和 字符串 無法拼接print(add(, '4,5,6'))
輸出如下
推薦下我本人原創的 《PyCharm 中文指南》電子書,內含大量(300張)的圖解,制作之精良,值得每個 Python 工程師點個收藏。