Writing Functions
Writing Functions
Docstrings
def count_letter(content, letter):
"""Count the number of times `letter` appears in `content`.
Args:
content (str): The string to search.
letter (str): The letter to search for.
Returns:
int
# Add a section detailing what errors might be raised
Raises:
ValueError: If `letter` is not a one-character string.
"""
if (not isinstance(letter, str)) or len(letter) != 1:
raise ValueError('`letter` must be a single character string.')
return len([char for char in content if char == letter])
如何直接获取
# Get the "count_letter" docstring by using an attribute of the function
docstring = count_letter.__doc__
border = '#' * 28
print('{}\n{}\n{}'.format(border, docstring, border))
<script.py> output:
############################
Count the number of times `letter` appears in `content`.
Args:
content (str): The string to search.
letter (str): The letter to search for.
Returns:
int
Raises:
ValueError: If `letter` is not a one-character string.
############################
inspect.getdoc()获取
data:image/s3,"s3://crabby-images/dce53/dce53e1b6c0d9d451e6c155c1884711833ecbcaf" alt="1654597449727.png"
Pass by assignment
和Java 的 Pass by reference相似。只不过需要注意不能更改的变量有什么。
data:image/s3,"s3://crabby-images/8562d/8562df886f7b10c6516b1ddf21a2586e4e2de923" alt="1654449339910.png"
data:image/s3,"s3://crabby-images/9b97c/9b97cb162127b1822a7064571c3d5760b49711da" alt="1654449307135.png"
context managers
作用
- 设置一个上下文
- 运行你的代码
- 删除上下文
open()
open() does three things:
- Sets up a context by opening a file
- Lets you run any code you want on that file
- Removes the context by closing the file
Using a context manager
with <context-manager>(<args>) as <variable-name>:
# Run your code here
# This code is running "inside the context"
# This code runs after the context is removed
举例
with open('my_file.txt')as my_file: text = my_file.read() length = len(text)
print('The file is {} characters long'.format(length))
define a context manager
Two ways:
- Class-based
- Function-based
此处关注基于函数的
data:image/s3,"s3://crabby-images/4cac6/4cac670394476c344b6f149fb69cc66522c8a48a" alt="1654597949891.png"
data:image/s3,"s3://crabby-images/170c3/170c36a11c269fccd739fdc703f3751df50f4321" alt="1654597982200.png"
应当添加try-catch 保证最终一定关闭资源
data:image/s3,"s3://crabby-images/9dcf4/9dcf4b56fdd92d44f9777194d037e08e4079aec7" alt="1654598055599.png"
Functions as objects
Python 思想:一切都是 object
调用和赋值
data:image/s3,"s3://crabby-images/6ab96/6ab9664c2c3db86af0fd14f4c6d783676e370704" alt="1654598199471.png"
Functions as variables
data:image/s3,"s3://crabby-images/6e164/6e164216c51413147caf5a06f366756ab7056605" alt="1654598284189.png"
子函数
def foo():
x = [3, 6, 9]
def bar(y):
print(y)
for value in x:
bar(x)
同样的,子函数对象也可以作返回值
data:image/s3,"s3://crabby-images/18e6b/18e6bce1ec3138d26799b0b8029439d539a8a03b" alt="1654598434996.png"
Scope
不妨先做个对比
Language | 全局变量 | 局部变量 | 全局变量在函数内 |
---|---|---|---|
C/C++ | 有(函数外声明) | 有(函数/{}内声明) | 可读 可写 |
Python3 | 有(函数外声明) (global函数内声明) | 有(函数内声明) | 可读, 不可写 加global后可写 |
Java | 无(只有类变量,被成员共享访问) | 有(函数/{}内声明) | 类变量可读可写 除非用关键字控制 |
Python 变量作用域
data:image/s3,"s3://crabby-images/e6bdc/e6bdcbd12b17eb412472bfc8ddfe37d57a3f0325" alt="1654601701804.png"
全局变量在函数内可读不可写
x = 7
y = 200
def foo():
x = 42 #创建x为局部变量,忽略全局变量x
print(x)
print(y)
foo()
42
200
global 关键字
global 关键字在函数内修饰变量,有两个作用
- 在函数内声明一个新的全局变量,并且使该变量在函数内可读可写
- 若修饰的变量已经在全局定义,则使该变量在函数内可读可写
data:image/s3,"s3://crabby-images/a7c84/a7c84eb00bb54400a312ea62e5a35fc40d46d3da" alt="1654599590138.png"
nonlocal 关键字
前面介绍了函数中可以创建子函数,然而子函数只能读取父函数的变量,并不能写。
nonlocal 关键字作用类似于global,是
在函数内声明一个新的变量,并且使该变量在其子函数内可读可写
data:image/s3,"s3://crabby-images/367b8/367b888e8b6fd7fc2dd9d41eb33f81bc409c2420" alt="1654600903405.png"
Closures
作用是 Attaching nonlocal variables to nested functions
思考子函数 离开了父函数 被传递给一个新的引用后,子函数 需要的读的父函数变量怎么获取?
答案就是 传递时,会将函数需要的 父函数变量 复制一份,放到 返回的子函数的__closure__
里面。
data:image/s3,"s3://crabby-images/5d856/5d85634fb8c5801e9b353b19756a2bb2d3cda256" alt="1654601087293.png"
这里有一点需要注意的是每一次将返回一个新的子函数object,而不是共享一个子函数object
因此__closure__
元组中的元素复制后就不会改变了。
删除变量不影响:
data:image/s3,"s3://crabby-images/4bce1/4bce16ea44c3b9031e801860aeda377a465321c8" alt="1654601507059.png"
修改变量不影响:
data:image/s3,"s3://crabby-images/8b096/8b09656fa0c4ac30e63d5a4456c6ade27f922c33" alt="1654601522306.png"
Decorators
首先明确一点,Java注解(Annotation) 和 Python的装饰器(Decorators)不是一个东西。Java也有Decorators。
data:image/s3,"s3://crabby-images/8d538/8d53804a9c752bfb28d97e7be114ba6d079a1b9f" alt="1654603831428.png"
Decorators 的功能总的来说就是劫持函数并修改其功能,实质是用一个新的函数把旧的函数进行一些修改后包装起来。
- 修改输入
- 修改输出
- 改变函数本身的行为
使用修饰器
@double_args
def multiply(a, b):
return a * b
multiply(1, 5)
20
创建修饰器
修饰器本身就是一个函数,以目标函数为参数,修改目标函数,之后把修改好的目标函数传传递给原引用。
data:image/s3,"s3://crabby-images/0953f/0953f3736b98e196cb69ae1daa2c67a0d3c1d7be" alt="1654604502166.png"
在创建修饰器的时候,需要注意的是应把修饰器函数的
- 接受唯一参数,即函数object
- 返回唯一参数,即函数object
- 返回的函数object的 参数个数以及类型顺序 应当和旧函数的 参数个数以及类型顺序 相同。
示例1
def print_before_and_after(func):
def wrapper(*args):
print('Before {}'.format(func.__name__))
# Call the function being decorated with *args
func(*args)
print('After {}'.format(func.__name__))
# Return the nested function
return wrapper
@print_before_and_after
def multiply(a, b):
print(a * b)
multiply(5, 10)
Before multiply
50
After multiply
示例2
def counter(func):
def wrapper(*args, **kwargs):
wrapper.count += 1
# Call the function being decorated and return the result
return func(*args, **kwargs)
wrapper.count = 0
# Return the new decorated function
return wrapper
# Decorate foo() with the counter() decorator
@counter
def foo():
print('calling foo()')
foo()
foo()
print('foo() was called {} times.'.format(foo.count))
calling foo()
calling foo()
foo() was called 2 times.
functools.wraps
由于被修饰后,原有的metadata将被包装而无法访问。
__name__
:函数的名称__defaults__
:函数的默认参数__doc__
:函数的注释
data:image/s3,"s3://crabby-images/2c8d2/2c8d22f98af9d44458bfa80b45fbe39a60c8591d" alt="1654608052881.png"
给子函数wrapper添加修饰器即可解决这个问题。这样新函数的所有metadata将会赋值包装前函数的元数据。
此外,加上这个修饰器后支持访问包装前的函数。使用 __wrapped__
data:image/s3,"s3://crabby-images/4fdd9/4fdd9f047d0563c5e6057ee2c5b4011aa2581892" alt="1654608201850.png"
Decorators Application
@timer
import time
def timer(func):
"""A decorator that prints how long a function took to run."""
# Define the wrapper function to return.
def wrapper(*args, **kwargs):
# When wrapper() is called, get the current time.
t_start = time.time()
# Call the decorated function and store the result.
result = func(*args, **kwargs)
# Get the total time it took to run, and print it.
t_total = time.time() - t_start
print('{} took {}s'.format(func.__name__, t_total))
return result
return wrapper
data:image/s3,"s3://crabby-images/e8387/e8387cbc14b3a692271a70a9ffec13bb683ddcd3" alt="1654607418130.png"
@memoize
def memoize(func):
"""Store the results of the decorated function for fast lookup
"""
# Store results in a dict that maps arguments to results
cache = {}
# Define the wrapper function to return.
def wrapper(*args, **kwargs):
# If these arguments haven't been seen before,
if (args, kwargs) not in cache:
# Call func() and store the result.
cache[(args, kwargs)] = func(*args, **kwargs)
# 注意如果cache中存在,则直接跳过了中间步骤而直接return
return cache[(args, kwargs)]
return wrapper
data:image/s3,"s3://crabby-images/a2583/a2583ecb544060cb51b200f21736fb35a394e46d" alt="1654607615207.png"
Decorators with args
run_n_times()
data:image/s3,"s3://crabby-images/6db54/6db549420b35f99fc1d55e05bfea0ab8d6b9a844" alt="1654626616735.png"
实现
不妨当做对旧的函数进行两次包装...
data:image/s3,"s3://crabby-images/894bd/894bd3656ac4ec5141693bec2966e53134d236c8" alt="1654627117384.png"
@timeout()
def timeout(n_seconds):
def decorator(func):
@wraps(func)
def wrapper(*args, **kwargs):
# Set an alarm for n seconds
signal.alarm(n_seconds)
try:
# Call the decorated func
return func(*args, **kwargs)
finally:
# Cancel alarm
signal.alarm(0)
return wrapper
return decorator
data:image/s3,"s3://crabby-images/0f7aa/0f7aa3ecfe62c7d22de0b84ea0f169e056e1b80e" alt="1654627288479.png"
@tag(*tags)
给某物打标签意味着你给该物打了一个或多个字符串,作为标签。例如,我们经常给电子邮件或照片打上标签,这样我们以后就可以搜索它们了。你决定写一个装饰器,让你用一个任意的标签列表来标记你的函数。你可以将这些标签用于许多事情。
def tag(*tags):
# Define a new decorator, named "decorator", to return
def decorator(func):
# Ensure the decorated function keeps its metadata
@wraps(func)
def wrapper(*args, **kwargs):
# Call the function being decorated and return the result
return func(*args, **kwargs)
wrapper.tags = tags
return wrapper
# Return the new decorator
return decorator
@tag('test', 'this is a tag')
def foo():
pass
print(foo.tags)
('test', 'this is a tag')
@returns()
def returns(return_type):
# Complete the returns() decorator
def decorator(func):
def wrapper(*args, **kwargs):
result = func(*args, **kwargs)
assert type(result) == return_type
return result
return wrapper
return decorator
@returns(dict)
def foo(value):
return value
try:
print(foo([1,2,3]))
except AssertionError:
print('foo() did not return a dict!')
foo() did not return a dict!