Python 小贴士(3)函数

19. 不要把函数返回的多个数值拆分到三个以上的变量中

拆包机制允许 Python 函数返回一个以上的值:

def get_stats(numbers):
    minimum = min(numbers)
    maximum = max(numbers)
    return minimum, maximum

lengths = [63, 73, 72, 60, 67, 66, 71, 61, 72, 70]

minimum, maximum = get_stats(lengths)

print(f'Min: {minimum}, Max: {maximum}')
>>>
Min: 60, Max: 73

在返回多个值的时候,可以用带星号的表达式接收那些没有被普通变量捕获到的值。

def get_avg_ratio(numbers):
    average = sum(numbers) / len(numbers)
    scaled = [x / average for x in numbers]
    scaled.sort(reverse=True)
    return scaled

longest, *middle, shortest = get_avg_ratio(lengths)

print(f'Longest:  {longest:>4.0%}')
print(f'Shortest: {shortest:>4.0%}')

>>>
Longest:  108%
Shortest:  89%

20. 遇到意外情况时应该抛出异常,不要返回 None

def careful_devide(a, b):
    try:
        return a / b
    except ZeroDivisionError:
        raise ValueError('Invalid inputs')

x, y = 5, 2
try:
    result = careful_devide(x, y)
except ValueError:
    print('Invalid inputs')
else:
    print('Result is %.1f' % result)

>>>
Result is 2.5

21. 了解如何在闭包里面使用外围作用域中的变量

Python 支持闭包,这让定义在大函数中的小函数也能应用大函数的变量。函数在 Python 中时一等对象,所以可以像操作其他对象那样直接引用它们、把它们赋给变量、将他们当成参数传给其他函数。Python 在判断两个序列(包括元组)的大小时,首先比较 0 号位置的两个元素,如果相等,就比较 1 号位置的两个元素,如果还相等,就比较 2 号位置的两个元素,依此类推,直到得出结论为止。

Python 有一种特殊的写法,可以把闭包中的数据赋给闭包外面的变量。

def sort_priority(numbers, group):
    found = False
    def helper(x):
        nonlocal found
        if x in group:
            found = True
            return (0, x)
        return (1, x)
    numbers.sort(key=helper)
    return found

22. 用数量可变的位置参数给函数设计清晰的参数列表

def log(message, *values):
    if not values:
        print(message)
    else:
        values_str = ', '.join(str(x) for x in values)
        print(f'{message}: {values_str}')

log('My numbers are', 1, 2)
log('Hi there')

>>>
My numbers are: 1, 2
Hi there

favorites = [7, 33, 99]
log('Favorite colors', *favorites)

>>>
Favorite colors: 7, 33, 99

23. 用关键字参数来表示可选的行为

def remainder(number, divisor):
    return number % divisor

# 以下四种效果相同
remainder(20, 7)
remainder(20, divisor=7)
remainder(number=20, divisor=7)
remainder(divisor=7, number=20)

如果位置参数和关键字参数混用,位置参数必须出现在关键字参数之前。

如果有一个字典,且 remainder 函数可以调用这个字典,那么可以把 ** 加在字典前面,这会让 Python 把字典里的键值对以关键字参数的形式传给函数。

my_kwargs = {
    'number': 20,
    'divisor': 7,
}
remainder(**my_kwargs)

my_kwargs = {
    'divisor': 7,
}
remainder(number=20, **my_kwargs)

my_kwargs = {
    'number': 20,
}
other_kwargs = {
    'divisor': 7,
}
remainder(**my_kwargs, **other_kwargs)

用关键字参数调用函数可以让初次阅读代码的人更容易看懂。可以带有默认值,该值是在定义函数时指定的。可以灵活地扩充函数的参数。

24. 用 None 和 docstring 来描述默认值会变的参数

def log(message, when=None):
    if when is None:
        when = datatime.now()
    print(f'{when}: {message}')
def decode(data, default=None):
    try:
        return json.loads(data)
    except ValueError:
        if default is None:
            default = {}
        return default

25. 用只能以关键字指定和只能按位置传入的参数来设计清晰的参数列表

def safe_division_e(numerator, denominator, /, ndigits=10, *, ignore_overflow=False, 
                    ignore_zero_division=False):
    try:
        fraction = numerator / denominator
        return round(fraction, ndigits)
    except OverflowError:
        if ignore_overflow:
            return 0
        else:
            raise
    except ZeroDivisionError:
        if ignore_zero_division:
            return float('inf')
        else:
            raise

在函数的参数列表中,/ 符号左侧的参数是只能按位置指定的参数,* 符号右侧的参数则是只能按关键字形式指定的参数。这两个符号之间的参数既可以按位置提供,又可以用关键字形式指定。

26. 用 functools.wraps 定义函数修饰器

这是个很有用的机制,能够确保用户以正确的方式使用函数,也能够用来调试程序或实现函数注册功能,此外还有许多用途。

from functools import wraps

def trace(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        result = func(*args, **kwargs)
        print(f'{func.__name__}({args!r}, {kwargs!r}) '
              f'-> {result!r}')
        return result
    return wrapper

@trace
def fibonacci(n):
    """Return the n-th Fibonacci number"""
    if n in (0, 1):
        return n
    return (fibonacci(n - 2) + fibonacci(n - 1))

help(fibonacci)

>>>
Help on function fibonacci in module __main__:

fibonacci(n)
    Return the n-th Fibonacci number
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值