3.3 返回值
函数并非总是直接显示输出,相反,它可以处理一些数据,并返回一个或一组值。在函数中,可使用return 语句将值返回到调用函数的代码行。
3.3.1 返回简单值
实例:函数,它接受名和姓并返回整洁的姓名。
def get_formatted_name(first_name,last_name):
'''返回格式化后的姓名'''
full_name = first_name + ' ' + last_name
return full_name
musician = get_formatted_name('connie','niu').title()
print(musician)
3.3.2 让实参变成可选
需要让实参变成可选的,这样使用函数的人就只需在必要时才提供额外的信息。可使用默认值来让实参变成可选的。例如:扩展函数get_formatted_name() ,使其还处理中间名。由于并非所有的人都有中间名,但如果你调用这个函数时只提供了名和姓,它将不能正确地运行。为让get_formatted_name() 在没有提供中间名时依然可行,可给实参middle_name 指定一个默认值---空字符串,并将其移到形参列表的末尾。
def get_formatted_name(first_name, last_name, middle_name=''):
"""返回整洁的姓名"""
if middle_name:
full_name = first_name + ' ' + middle_name + ' ' + last_name
else:
full_name = first_name + ' ' + last_name
return full_name.title()
musician = get_formatted_name('jimi', 'hendrix')
print(musician)
musician = get_formatted_name('john', 'hooker', 'lee')
print(musician)
示例详细说明:
姓名是根据名、姓、中间名三个可能提供的部分创建的。由于人都有名和姓,因此在函数定义中首先列出了这两个形参。中间名是可选的,因此在函数定义中最后列出该形参,并将其默认值设置为空字符串(再次强调:使用参数使用默认值,多个传递时是按照位置参数解析,即严格依据参数顺序);
在函数体中,检查是否提供了中间名。Python将非空字符串解读为True ,因此如果函数调用中提供了中间名,if middle_name 将为True判断;
如果提供了中间名,就将名、中间名和姓合并为姓名;
如果没有提供中间名,middle_name 将为空字符串,导致if 测试未通过,进而执行else 代码块,只使用名和姓来生成姓名;
然后将其修改为首字母大写格式,并返回到函数调用行;
在函数调用行,将返回的值存储在变量musician 中;然后将这个变量的值打印出来。
3.3.3 返回字典
函数可返回任何类型的值,包括列表和字典等较复杂的数据结构。
实例:定义函数接受姓名的组成部分,并返回一个表示人的字典。
def build_person(first_name, last_name):
"""返回一个字典,其中包含有关一个人的信息"""
person = {'first': first_name, 'last': last_name}
return person
musician = build_person('jimi', 'hendrix')
print(musician)
3.4 传递列表
向函数传递列表很有用,这种列表包含的可能是名字、数字或更复杂的对象(如字典)。将列表传递给函数后,函数就能直接访问其内容。
实例:假设有一个用户列表,将列表传递给一个名为greet_users() 的函数,在函数中问候列表中的每个人。
def greet_users(names):
"""向列表中的每位用户都发出简单的问候"""
for name in names:
msg = "Hello, " + name.title() + "!"
print(msg)
usernames = ['hannah', 'ty', 'margot']
greet_users(usernames)
3.4.1 在函数中修改列表
将列表传递给函数,在函数中对这个列表所做的任何修改都是永久性的,可高效地处理大量的数据。
def print_models(unprinted_designs, completed_models):
"""
模拟打印每个设计,直到没有未打印的设计为止
打印每个设计后,都将其移到列表completed_models中
"""
while unprinted_designs:
current_design = unprinted_designs.pop()
#模拟根据设计制作3D打印模型的过程
print("Printing model: " + current_design)
completed_models.append(current_design)
def show_completed_models(completed_models):
"""显示打印好的所有模型"""
print("\nThe following models have been printed:")
for completed_model in completed_models:
print(completed_model)
unprinted_designs = ['iphone case', 'robot pendant', 'dodecahedron']
completed_models = []
print_models(unprinted_designs,completed_models)
show_completed_models(completed_models)
示例详解:
我们定义了函数print_models() ,它包含两个形参:一个需要打印的设计列表和一个打印好的模型列表。给定这两个列表,这个函数模拟打印每个设计的过程:将设计逐个地从未打印的设计列表中取出,并加入到打印好的模型列表中。
我们定义了函数show_completed_models() ,它包含一个形参:打印好的模型列表。给定这个列表,函数show_completed_models() 显示打印出来的每个模型的名称。
接下来为主程序,创建一个未打印的设计列表,还创建了一个空列表,用于存储打印好的模型。调用print_models() 并向它传递两个列表,并完成相关处理;
调用show_completed_models() ,并将打印好的模型列表传递给它,让其能够指出打印了哪些模型。
这个程序还演示了这样一种理念,即每个函数都应只负责一项具体的工作。第一个函数打印每个设计,而第二个显示打印好的模型;这优于使用一个函数来完成两项工作。编写函数时,如果你发现它执行的任务太多,请尝试将这些代码划分到两个函数中。别忘了,总是可以在一个函数中调用另一个函数,这有助于将复杂的任务划分成一系列的步骤。这样组织代码让程序更容易扩展和维护。
3.4.2 禁止函数修改列表
有时候,需要禁止函数修改列表。例如,像前一个示例那样,程序执行完成,未打印的设计列表会被清空,若希望保留原来的未打印的设计列表,以供备案。则可将列表的副本传递给函数:print_models(unprinted_designs[:], completed_models)
函数print_models()使用的是列表unprinted_designs 的副本,而不是列表本身。
虽然向函数传递列表的副本可保留原始列表的内容,但除非有充分的理由需要传递副本,否则还是应该将原始列表传递给函数,因为会消耗更多的时间和内存。