【Python学习】七、函数

7.1 定义函数 

def greet_user():
    """向用户问好"""
    print("Hello user!")

greet_user()

第一行的代码行使用关键字def 来告诉Python你要定义一个函数。

这是函数定义 ,向Python指出了函数名, 还可能在括号内指出函数为完成其任务需要什么样的信息。在这里,函数名为greet_user() ,它不需要任何信息就能完成其工作,因此括号是空的(即便如此,括号也必不可少)。最后,定义以冒号结尾。

第二行的文本是被称为文档字符串 (docstring)的注释,描述了函数是做什么的。文档字符串用三引号括起,Python使用它们来生成有关程序中函数的文档。

第三行代码行是函数体内的唯一一行代码

要使用这个函数,可调用它。函数调用让Python执行函数的代码。要调用函数,可依次指定函数名以及用括号括起的必要信息,如第四行。

 7.1.1 向函数传递信息

以上一个函数为例,为了向函数传递信息,只需在括号内添加一个变量username,就可让函数接 受你给username指定的任何值。

def greet_user(username):
    """向用户问好"""
    print("Hello " + username.title() + ".")

greet_user('alice')

7.1.2 形参和实参

在函数greet_user() 的定义中,变量username 是一个形参 ——函数完成其工作所需的一项信息。

在代码greet_user('alice') 中, 值'alice'是一个实参 。实参是调用函数时传递给函数的信息。

在greet_user('alice') 中,将实参'alice' 传递给了函 数greet_user() ,这个值被存储在形参username 中。

7.2 传递实参

7.2.1 位置实参

你调用函数时,Python必须将函数调用中的每个实参都关联到函数定义中的一个形参。为此,最简单的关联方式是基于实参的顺序。这种关联方式被称为位置实参 。

def describe_pet(pet_kind, pet_name):
    """描述一个动物的类型和名称"""
    print("I have a " + pet_kind + " called " + pet_name +".")

describe_pet('dog', 'Tom')

 位置实参的位置特别重要!如果不按照次序对应,则会出现离谱的结果。

7.2.2 关键字实参

关键字实参是传递给函数的名称—值对。

你直接在实参中将名称和值关联起来了,因此向函数传递实参时不会混淆(不会得到名为Hamster 的harry这样的结果)。

关键字实参让你无需考虑函数调用中的实参顺序,还清楚地指出了函数调用中各个值的用途。

def describe_pet(pet_kind, pet_name):
    """描述一个动物的类型和名称"""
    print("I have a " + pet_kind + " called " + pet_name +".")

describe_pet('dog', 'Tom')
describe_pet(pet_kind='cat', pet_name='Bob')

 关键字实参的顺序无关紧要,因为Python知道各个值该存储到哪个形参中。

7.2.3 默认值

编写函数时,可给每个形参指定默认值 。

在调用函数中给形参提供了实参时,Python将使用指定的实参值;

否则,将使用形参的默认值。

因此,给形参指定默认值后,可在函数调用中省略相应的实参。

使用默认值可简化函数调用,还可清楚地指出函数的典型用法。

def describe_pet(pet_name, pet_kind='dog'):
    """描述一个动物的类型和名称"""
    print("I have a " + pet_kind + " called " + pet_name +".")

describe_pet('Frank')

这里修改了函数describe_pet() 的定义,在其中给形参pet_kind指定了默认值'dog' 。这样,调用这个函数时,如果没有给pet_kind指定值,Python将把这个形参设置为'dog' 

请注意,在这个函数的定义中,修改了形参的排列顺序。由于给 pet_kind指定了默认值,无需通过实参来指定动物类型,因此在 函数调用中只包含一个实参——宠物的名字。然而,Python依然将这个实参视为位置实参,因此如果函数调用中只包含宠物的名字,这个实参 将关联到函数定义中的第一个形参。这就是需要将pet_name放在形参列表开头的原因所在。 

def describe_pet(pet_name, pet_kind='dog'):
    """描述一个动物的类型和名称"""
    print("I have a " + pet_kind + " called " + pet_name +".")

describe_pet('Frank')
describe_pet("Frank", "cat")

注意 :使用默认值时,在形参列表中必须先列出没有默认值的形参,再列出有默认值的实参。这让Python依然能够正确地解读位置实参。

7.3 返回值

函数并非总是直接显示输出,相反,它可以处理一些数据,并返回一个或一组值。函数返回的值被称为返回值 。在函数中,可使用return 语句将值返回到调用函数的代码行。

7.3.1 返回简单返回值

def get_formatted_name(first_name, last_name):
    full_name = first_name.title() + ' ' + last_name.title()
    return full_name

name = get_formatted_name('chow', 'runfat')
print(name)

7.3.2 让参数变成可选的

有时候,需要让实参变成可选的,这样使用函数的人就只需在必要时才 提供额外的信息。可使用默认值来让实参变成可选的。

def get_formatted_name(first_name, last_name, middle_name = ' '):
    full_name = (first_name + " " + middle_name + " " + last_name).title()
    return full_name

name = get_formatted_name('chow', 'fat', 'run')
print(name)
name = get_formatted_name('wu', 'ke')
print(name)

这个修改后的版本适用于只有名和姓的人,也适用于还有中间名的人。

7.3.3 返回词典

函数可返回任何类型的值,包括列表和字典等较复杂的数据结构。例 如,下面的函数接受姓名的组成部分,并返回一个表示人的字典.

def build_person(first_name, last_name):
    person = {'first' : first_name.title(), 'last' : last_name.title()}
    return  person

p1 = build_person('chow', 'runfat')
print(p1)

你可以轻松地扩展这个函数,使其接受可选值,如中间名、年龄、职业或你要存储的其他任何信 息。例如,下面的修改让你还能存储年龄:

def build_person(first_name, last_name, age = ''):
    person = {'first' : first_name.title(), 'last' : last_name.title()}
    if age:
        person['age'] = age
    return person

p1 = build_person('chow', 'runfat', str(32))
print(p1)

 7.3.4 结合while循环

可将函数同本书前面介绍的任何Python结构结合起来使用。例如,下面 将结合使用函数get_formatted_name() 和while 循环,以更正规的方式问候用户。

def get_formatted_name(first_name, last_name, middle_name=' '):
    full_name = (first_name + " " + middle_name + " " + last_name).title()
    return full_name

while True:
    print("please key in your first name and last name:")
    print('\n key in quit to exit.')
    f_name = input("please key in your first name:")
    if f_name == 'q':
        break
    l_name = input('please key in your last name:')
    if l_name == 'q':
        break
    name = get_formatted_name(f_name, l_name)
    print('Hello ' + name + "!")

7.4 传递列表

你经常会发现,向函数传递列表很有用,这种列表包含的可能是名字、 数字或更复杂的对象(如字典)。将列表传递给函数后,函数就能直接访问其内容。下面使用函数来提高处理列表的效率。

def greet_users(names):
    for name in names:
        print("Hello " + name.title() + '!')

user_names = ['alice', 'bob', 'carol', 'dave']
greet_users(user_names)

 7.4.1 在函数中修改列表

将列表传递给函数后,函数就可对其进行修改。在函数中对这个列表所做的任何修改都是永久性的,这让你能够高效地处理大量的数据。

# 首先创建一个列表,其中包含一些要打印的设计
unprinted_designs = ['iphone case', 'robot pendant', 'dodecahedron']
completed_models = []
# 模拟打印每个设计,直到没有未打印的设计为止
# 打印每个设计后,都将其移到列表completed_models中
while unprinted_designs:
    current_design = unprinted_designs.pop()
#模拟根据设计制作3D打印模型的过程
    print("Printing model: " + current_design)
    completed_models.append(current_design)
# 显示打印好的所有模型
print("\nThe following models have been printed:")
for completed_model in completed_models:
    print(completed_model)

 为重新组织这些代码,我们可编写两个函数,每个都做一件具体的工 作。大部分代码都与原来相同,只是效率更高。第一个函数将负责处理打印设计的工作,而第二个将概述打印了哪些设计

def print_design(unprinted_designs, completed_models):
    while unprinted_designs:
        current_design = unprinted_designs.pop()
        print("Printing model: " + current_design)
        completed_models.append(current_design)

def already_design(completed_models):
    print("\nThe following models have been printed:")
    for completed_model in completed_models:
        print(completed_model)

up_designs = ['apple', 'bmw', 'huawei']
cp_designs = []
print_design(up_designs, cp_designs)
already_design(cp_designs)

这个程序的输出与未使用函数的版本相同,但组织更为有序。完成大部 分工作的代码都移到了两个函数中,让主程序更容易理解。

7.4.2 禁止函数修改列表

有时候,需要禁止函数修改列表,为此可以向函数传递列表的副本而不是原件;这样函数所做的任何修改都只影响副本,而丝毫不影响原件。

def print_design(unprinted_designs, completed_models):
    while unprinted_designs:
        current_design = unprinted_designs.pop()
        print("Printing model: " + current_design)
        completed_models.append(current_design)

def already_design(completed_models):
    print("\nThe following models have been printed:")
    for completed_model in completed_models:
        print(completed_model)

up_designs = ['apple', 'bmw', 'huawei']
cp_designs = []
print_design(up_designs[:], cp_designs)
already_design(cp_designs)

print("\n")
for up_design in up_designs:
    print(up_design)

8.5 传递任意数量的实参

有时候,你预先不知道函数需要接受多少个实参,好在Python允许函数从调用语句中收集任意数量的实参。

下面的函数只有一个形参*toppings ,但不管调用语句提供了多少实参,这个形参都将它们统统收入囊中:

def pizza(*toppings):
        print(toppings)

pizza('apple')
pizza('apple', 'banana', 'orange')

def pizza(*toppings):
    print("我们将在披萨中加如下配料:")
    for topping in toppings:
        print(topping)

pizza('apple')
pizza('apple', 'banana', 'orange')

7.5.1 结合使用位置实参和任意数量实参

如果要让函数接受不同类型的实参,必须在函数定义中将接纳任意数量实参的形参放在最后。Python先匹配位置实参和关键字实参,再将余下的实参都收集到最后一个形参中。

def pizza(size, *toppings):
    print("我们要制作一个" + str(size) +"寸大小的披萨。")
    print("我们将在披萨中加如下配料:")
    for topping in toppings:
        print(topping)

pizza(5, 'apple')
pizza(12, 'apple', 'banana', 'orange')

 7.5.2 使用任意数量的关键字实参

有时候,需要接受任意数量的实参,但预先不知道传递给函数的会是什 么样的信息。在这种情况下,可将函数编写成能够接受任意数量的键—值对——调用语句提供了多少就接受多少。

def build_profile(first, last, **info):
    person = {}
    person['first_name'] = first
    person['last_name'] = last
    for key, value in info.items():
        person[key] = value
    return person

user1 = build_profile('chung', 'runfate', home = 'hongkong', school = 'PKU')
print(user1)

调用这个函数时,不管额外提供了多少个键—值对,它都能正确地处理。 

7.6 将函数存储在模块中

函数的优点之一是,使用它们可将代码块与主程序分离。

你还可以更进一步,将函数 存储在被称为模块的独立文件中,再将模块导入到主程序中。

import 语句允许在当前运行的程序文件中使用模块中的代码。

7.6.1 导入整个模块

要让函数是可导入的,得先创建模块。

模块是扩展名为.py的文件,包含要导入到程序中的代码。

# pizza.py
def make_pizza(size, *toppings):
    print("我们将要制作" + str(size) + "寸的披萨。")
    for topping in toppings:
        print("我们将要在披萨里增加" + topping + "配料.")
# 主程序
import pizza

pizza.make_pizza(12, 'apple', 'orange', 'beef', 'meat')

代码行import pizza 让Python打开文件 pizza.py,并将其中的所有函数都复制到这个程序中。

要调用被导入的模块中的函数,可指定导入的模块的名称pizza 和函数名make_pizza() ,并用句点分隔它们。

7.6.2 导入特定的函数 

你还可以导入模块中的特定函数,这种导入方法的语法如下

from module_name import function_name
from module_name import function_0, function_1, function_2
from pizza import make_pizza

make_pizza(12, 'apple', 'orange', 'beef', 'meat')

 7.6.3 使用as给函数指定别名

如果要导入的函数的名称可能与程序中现有的名称冲突,或者函数的名称太长,可指定简短而独一无二的别名——函数的另一个名称,类似于外号。

要给函数指定这种特殊外号,需要在导入它时这样做。

from pizza import make_pizza as mp

mp(12, 'apple', 'orange', 'beef', 'meat')

 指定别名的通用语法如下:

from module_name import function_name as fn

7.6.4 使用as 给模块指定别名

你还可以给模块指定别名。通过给模块指定简短的别名

import module_name as mn
import pizza as p

p.make_pizza(12, 'apple', 'orange', 'beef', 'meat')

 7.6.5 导入模块中的所有函数

使用星号(* )运算符可让Python导入模块中的所有函数

from pizza import *
make_pizza(12, 'apple', 'orange', 'beef', 'meat')

import 语句中的星号让Python将模块pizza中的每个函数都复制到这个程序文件中。

由于导入了每个函数,可通过名称来调用每个函数,而无需使用句点表示法。

然而,使用并非自己编写的大型模块时,最好不要采用这种导入方法:如果模块中有函数的名称与你的项目中使用的名 称相同,可能导致意想不到的结果:Python可能遇到多个名称相同的函数或变量,进而覆盖函数,而不是分别导入所有的函数。

最佳的做法是,要么只导入你需要使用的函数,要么导入整个模块并使 用句点表示法。这能让代码更清晰,更容易阅读和理解。

7.7 练习

7-1 消息 :编写一个名为display_message() 的函数,它打印一 个句子,指出你在本章学的是什么。调用这个函数,确认显示的消息正确无误。

# 7-1
def display_message():
    print("今天我们学习函数。")

display_message()

7-2 喜欢的图书 :编写一个名为favorite_book() 的函数,其中 包含一个名为title的形参。这个函数打印一条消息,如One of my favorite books is Alice in Wonderland 。调用这个函 数,并将一本图书的名称作为实参传递给它。

# 7-2
def favourite_book(title):
    print("One of my favorite books is " + title.title() + ".")

favourite_book('the three kingdoms')

7-3 T恤 :编写一个名为make_shirt() 的函数,它接受一个尺码以及要印到T恤上的字样。这个函数应打印一个句子,概要地说明T恤的尺码和字样。

# 7-3
def make_shirt(size, pattern):
    print("印制一款" + str(size) + '尺寸并且有' + pattern + "图案的T恤。")

make_shirt(12, 'pku')
make_shirt(size=12, pattern='pku')

7-4 大号T恤 :修改函数make_shirt() ,使其在默认情况下制作 一件印有字样“I love Python”的大号T恤。调用这个函数来制作如下T恤:一件印有默认字样的大号T恤、一件印有默认字样的中号T恤 和一件印有其他字样的T恤(尺码无关紧要)。

# 7-4
def make_shirt(size, pattern='I love python!'):
    print("印制一款" + str(size) + '尺寸并且有' + pattern + "图案的T恤。")

make_shirt('大号', )
make_shirt('中号', )
make_shirt('中号', 'China')

7-5 城市 :编写一个名为describe_city() 的函数,它接受一座城市的名字以及该城市所属的国家。这个函数应打印一个简单的句子,如Reykjavik is in Iceland 。给用于存储国家的形参指定默认值。为三座不同的城市调用这个函数,且其中至少有一座城市不属于默认国家。

# 7-5
def describe_city(city='Reykjavik', country='Iceland'):
    print(city.title() + " is in " + country.title() + '.')

describe_city('taipei', 'china')
describe_city('hongkong', 'china')
describe_city('london', 'britain')

7-6 城市名 :编写一个名为city_country() 的函数,它接受城市 的名称及其所属的国家。

# 7-6
def city_country(city, country):
    message = city.title() + ',' + country.title()
    return message

print(city_country('taipei', 'china'))
print(city_country('jinan', 'china'))
print(city_country('tokyo', 'japan'))

7-7 专辑 :编写一个名为make_album() 的函数,它创建一个描述音乐专辑的字典。这个函数应接受歌手的名字和专辑名,并返回一个包含这两项信息的字典。使用这个函数创建三个表示不同专辑的字典,并打印每个返回的值,以核实字典正确地存储了专辑的信息。

给函数make_album() 添加一个可选形参,以便能够存储专辑包含的歌曲数。如果调用这个函数时指定了歌曲数,就将这个值添加到表示专辑的字典中。调用这个函数,并至少在一次调用中指定专辑包含的歌曲数。

# 7-7
def make_album(singer, album):
    dict = {'singer': singer, 'albunm': album}
    return dict

print(make_album('Jay chow', 'Qi Li Xiang'))
print(make_album('Zheng Zhihua', 'Xing Xing Dian Deng'))
print(make_album('Ren Xian Qi', 'Tian Ya'))

def make_album(singer, album, number =''):
    dict = {'singer': singer, 'albunm': album}
    if number:
        dict['number'] = number
    return dict

print(make_album('Ren Xian Qi', 'Tianya', 9))

 7-8 用户的专辑 :在为完成练习8-7编写的程序中,编写一个while循环,让用户输入一个专辑的歌手和名称。获取这些信息后,使用它们来调用函数make_album() ,并将创建的字典打印出来。在这 个while 循环中,务必要提供退出途径。

# 7-8

def make_album(singer, album, number =''):
    dict = {'singer': singer, 'albunm': album}
    if number:
        dict['number'] = number
    return dict

active = True
while active:
    singer = input("请输入歌手名,退出请按q。")
    if singer == 'q':
        break
    album = input('请输入专辑名,退出请按q。')
    if album == 'q':
        break
    number = input("(可选)请输入数目,退出请按q。")
    if number == 'q':
        break
    print(make_album(singer, album, number))

7-9 魔术师 :创建一个包含魔术师名字的列表,并将其传递给一个 名为show_magicians() 的函数,这个函数打印列表中每个魔术师的名字。

# 7-9
def show_magicians(*names):
    for name in names:
        print(name)

show_magicians('zsc', 'ltt', 'zm', 'dq', 'zcl', 'ygz')

 7-10 了不起的魔术师 :在你为完成练习8-9而编写的程序中,编写一个名为make_great() 的函数,对魔术师列表进行修改,在每个魔术师的名字中都加入字样“the Great”。调用函数show_magicians() ,确认魔术师列表确实变了。

# 7-10
def show_magicians(names):
    for name in names:
        print(name)

def make_great(names):
    modified_names = []
    while names:
        current_name = "The Great " +names.pop()
        modified_names.append(current_name)
    return modified_names

names = ['zsc', 'ltt', 'zm', 'dq', 'zcl', 'ygz']
show_magicians(names)
names = make_great(names)
show_magicians(names)

 7-11 不变的魔术师 :修改你为完成练习8-10而编写的程序,在调 用函数make_great() 时,向它传递魔术师列表的副本。由于不想 修改原始列表,请返回修改后的列表,并将其存储到另一个列表 中。分别使用这两个列表来调用show_magicians() ,确认一个列 表包含的是原来的魔术师名字,而另一个列表包含的是添加了字 样“the Great”的魔术师名字。

def show_magicians(names):
    for name in names:
        print(name)

def make_great(names):
    modified_names = []
    while names:
        current_name = "The Great " +names.pop()
        modified_names.append(current_name)
    return modified_names

names =  ['zsc', 'ltt', 'zm', 'dq', 'zcl', 'ygz']
show_magicians(names)
print("\n")
names_new = make_great(names[:])
show_magicians(names)
print('\n')
show_magicians(names_new)
print('\n')

 7-12 三明治 :编写一个函数,它接受顾客要在三明治中添加的一系列食材。这个函数只有一个形参(它收集函数调用中提供的所有 食材),并打印一条消息,对顾客点的三明治进行概述。调用这个函数三次,每次都提供不同数量的实参。

# 7-12
def make_sandwich(*toppings):
    for topping in toppings:
        print(topping)

make_sandwich('apple')
print("\n")
make_sandwich('apple', 'banana')
print("\n")
make_sandwich('apple','banana', 'orange')
print("\n")

7-13 用户简介 :复制前面的程序user_profile.py,在其中调 用build_profile() 来创建有关你的简介;调用这个函数时,指 定你的名和姓,以及三个描述你的键-值对。

# 7-13
def build_profile(first, last, **info):
    profile = {}
    profile['first name'] = first
    profile['last name'] = last
    for key, value in info.items():
        profile[key] = value
    return profile

my_info = build_profile("Lee", "Ming",home='Beijing', age='34', school="PKU")

print(my_info)

7-14 汽车 :编写一个函数,将一辆汽车的信息存储在一个字典 中。这个函数总是接受制造商和型号,还接受任意数量的关键字实参。这样调用这个函数:提供必不可少的信息,以及两个名称—值 对,如颜色和选装配件。

# 7-14
def car_infos(brand, number, **infos):
    car = {}
    car['brand'] = brand
    car['number'] = number
    for key, value in infos.items():
        car[key] = value
    return car

car1 = car_infos('BMW', 'Z7', color='white', country='China', age=3)
print(car1)

 

 7-16 导入 :选择一个你编写的且只包含一个函数的程序,并将这 个函数放在另一个文件中。在主程序文件中,使用下述各种方法导 入这个函数,再调用它:

# add.py
def add2(num1, num2):
    return  num1 + num2
# main

import add
print(add.add2(1,3))

from add import add2
print(add2(1,3))

import add as ad
print(ad.add2(1,3))

from add import add2 as a2
print(a2(1,3))

from add import  *
print(add2(1,3))

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值