python基础模块---functools

import functools

'''
functools.cmp_to_key(func)
    将老式的比较函数(func)转换为关键字函数(key function)。在 Python3中比较大小、排序都是基于关键字函数的,
    Python 3 不支持老式的比较函数。
'''

class Person(object):

    def __init__(self, name):
        self.name = name

    def __repr__(self):
        return f"My name is {self.name}"


def compare(p1, p2):
    return len(p1.name) - len(p2.name)


data = [Person('张三'), Person('赵飞燕'), Person('汉武大帝'), Person('松下幸之助'), Person('爱新觉罗玄烨'), Person('阿尔伯特爱因斯坦')]
data.sort(key=functools.cmp_to_key(compare))
print(data)
# [My name is 张三, My name is 赵飞燕, My name is 汉武大帝, My name is 松下幸之助, My name is 爱新觉罗玄烨, My name is 阿尔伯特爱因斯坦]


'''
functools.reduce(function, iterable[, initializer])
    将初始值(默认为 0,可由 initializer 参数指定)、迭代器的当前元素传入 function 函数,将计算出来的函数结果作为下一次计算的初始值、
迭代器的下一个元素再次调用 function 函数……依此类推,直到迭代器的最后一个元素。
'''
print(functools.reduce(lambda x, y: x + y, range(10), 100))

'''
@functools.lru_cache(maxsize=128, typed=False)
    该函数装饰器使用 LRU(最近最少使用)缓存算法来缓存相对耗时的函数结果,避免传入相同的参数重复计算。同时,缓存并不会无限增长,
不用的缓存会被释放。其中 maxsize 参数用于设置缓存占用的最大字节数,typed 参数用于设置将不同类型的缓存结果分开存放。
'''


@functools.lru_cache(maxsize=128)
def fibnacci(n):
    if n in (1, 2):
        return 1
    else:
        return fibnacci(n - 1) + fibnacci(n - 2)


print(fibnacci(100))


'''
functools.partial(func, *args, **keywords)
    该函数用于为func函数的部分参数指定参数值,从而得到一个转换后的函数,程序以后调用转换后的函数时,就可以少传入那些己指定值的参数。
'''


def add(x, y):
    return x + y


par_func = functools.partial(add, 10)
print(par_func(5))  # 15


'''
partialmethod() 与 partial() 函数的作用基本相似,区别只是 partial() 函数用于为函数的部分参数绑定值;
而 partialmethod() 函数则用于为类中方法的部分参数绑定值。如下程序示范了 partialmethod() 函数的用法:
'''

class Person(object):

    def __init__(self):
        self._age = 18

    @property
    def age(self):
        return self._age

    def set_age(self, number: int):
        if not isinstance(number, int):
            raise ValueError('int needed')
        if number > 60:
            self._age = 100
        elif number < 10:
            self._age = 5
        else:
            self._age = 18

    set_young = functools.partialmethod(set_age, 8)
    set_old = functools.partialmethod(set_age, 70)


p = Person()
print(p.age)        # 18
p.set_young()       # 相当于调用set_age(8)
print(p.age)        # 5
p.set_old()         # 相当于调用set_age(70)
print(p.age)        # 100


'''
@functools.total_ordering
    这个类装饰器(作用类似于函数装饰器,只是它用于修饰类)用于为类自动生成比较方法。通常来说,开发者只要提供 
__lt__()、__le__()、__gt__()、__ge__() 其中之一(最好能提供 __eq__() 方法),@functools.total_ordering装饰器就会
为该类生成剩下的比较方法。
'''

@functools.total_ordering
class Person(object):

    def __init__(self, age):
        self.age = age

    def __eq__(self, other):
        return self.age == other.age

    def __gt__(self, other):
        if self.age > other.age:
            return True
        return False


p1 = Person(18)
p2 = Person(20)
print(p1 > p2)      # False
print(p1 >= p2)     # False
print(p1 < p2)      # True
print(p1 <= p2)     # True
print(p1 == p2)     # False


'''
@functools.singledispatch
    该函数装饰器用于实现函数对多个类型进行重载。比如同样的函数名称,为不同的参数类型提供不同的功能实现.
    该函数的本质就是根据参数类型的变换,将函数转向调用不同的函数。
'''


@functools.singledispatch
def test(args):
    print(args)


@test.register(int)
def test_int(args):
    print('the args is int')


@test.register(float)
def test_float(args):
    print('the args is float')


@test.register(list)
def test_list(args):
    print('the args is list')


@test.register(dict)
@test.register(set)
def test_dict_set(args):
    print('the args is dict or set')


test(1)     # the args is int
test(1.0)       #the args is float
test(list(range(5)))  # the args is list
test(dict([(1, 2), (3, 4)]))  # the args is dict or set
print(test.registry.keys())  # 获取test函数所绑定的全部类型
print(test.registry[int])  # 获取test函数绑定的int类型的函数

'''
@functools.wraps(wrapped, assigned=WRAPPER_ASSIGNMENTS, updated=WRAPPER_UPDATES)
    该函数装饰器用于修饰包装函数,使包装函数看上去就像 wrapped 函数。
'''


def decorator(func):
    @functools.wraps(func)
    def wrapper(*args, **kwargs):
        result = func(*args, **kwargs)
        return result

    return wrapper


@decorator
def test():
    """我是test函数"""
    print('test is calling')


test()  # test is calling
print(test.__name__)  # test,不加装饰器会显示为wrapper
print(test.__doc__)  # 我是test函数,不加装饰器得不到说明文档


'''
    functools.update_wrapper和functools.wraps的区别在于,functools.update_wrapper是一个函数,它的第一个参数是包装函数,第二个参数是
被包装函数。
'''
def decorator(func):

    def wrapper(*args, **kwargs):
        result = func(*args, **kwargs)
        functools.update_wrapper(wrapper, func)
        return result

    return wrapper

@decorator
def test():
    """我是test函数"""
    print('test is calling')

test()
print(test.__name__)
print(test.__doc__)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容