函数的基本写法如下所示:
def function_name(parameter1, parameter2, ...): """ Docstring: 描述函数的功能、参数和返回值 (可选但强烈推荐) """ # 函数体: 实现功能的代码 # ... return value # 可选,用于返回结果
# 定义一个简单的问候函数
def greet():
"""打印一句问候语。"""
message = "大家好!欢迎学习Python函数定义!"
print(message)
greet()
大家好!欢迎学习Python函数定义!
可以看到上述函数没有参数,也没有返回值,调用这个函数的时候运行函数体中的内容
# 查看文档字符串,方便查看函数的使用,这个方法可以不掌握
print(greet.__doc__)
# 实际上,当你在py文件中,鼠标悬停在函数上按住ctrl即可点击函数跳转到其内部查看函数的定义
打印一句问候语。
带上注释是函数定义的良好习惯,尤其在复杂的项目中,更方便维护和调试。可以让ai给你补上注释
函数的参数我们有如下称呼:
注意点: 定义的时候把函数的参数称之为形参,调用的时候把函数的参数称之为实参。
# 定义一个带一个参数的问候函数
def greet_person(name):
"""根据给定的名字打印问候语。
Args:
name (str): 要问候的人的名字。
"""
message = f"你好, {name}! 很高兴认识你。"
print(message)
greet_person("张三") # 输出: 你好, 张三! 很高兴认识你。
你好, 张三! 很高兴认识你。
# 定义一个带多个参数的函数 (例如,在机器学习中计算两个特征的和)
def add_features(feature1, feature2):
"""计算两个数值特征的和。
Args:
feature1 (float or int): 第一个特征值。
feature2 (float or int): 第二个特征值。
"""
total = feature1 + feature2
print(f"{feature1} + {feature2} = {total}")
add_features(10, 25) # 输出: 10 + 25 = 35
10 + 25 = 35
# add_features(5)
# 这会报错,因为少了一个参数 TypeError
函数不仅可以执行操作(如打印),还可以计算并返回一个结果
# 定义一个计算和并返回结果的函数
def calculate_sum(a, b):
"""计算两个数的和并返回结果。
Args:
a (float or int): 第一个数。
b (float or int): 第二个数。
Returns:
float or int: 两个数的和。
"""
result = a + b
return result
print("hhh")
calculate_sum(2, 3)
5
此时,注意到,print("hhh")这个代码并没有被执行,因为函数在遇到return语句时,就会立即返回,而不会继续执行函数后面的代码。
其次,如果没有return语句,或者return后面不带任何参数,那么函数也会返回None(不要把执行的操作理解为返回值)。
# 函数可以返回多种类型的数据,包括列表、字典等
# 例如,在数据预处理中,一个函数可能返回处理后的特征列表
def preprocess_data(raw_data_points):
"""模拟数据预处理,例如将所有数据点乘以2。
Args:
raw_data_points (list): 原始数据点列表。
Returns:
list: 处理后的数据点列表。
"""
processed = []
for point in raw_data_points:
processed.append(point * 2) # 假设预处理是乘以2
return processed
data = [1, 2, 3, 4, 5]
processed_data = preprocess_data(data)
print(f"原始数据: {data}")
print(f"预处理后数据: {processed_data}") # 输出: [2, 4, 6, 8, 10]
原始数据: [1, 2, 3, 4, 5] 预处理后数据: [2, 4, 6, 8, 10]
理解变量在何处可见和可访问非常重要。
局部变量 (Local Variables): 在函数内部定义的变量,只在该函数内部有效。当函数执行完毕后,局部变量通常会被销毁。
全局变量 (Global Variables): 在所有函数外部定义的变量,可以在程序的任何地方被访问(但在函数内部修改全局变量需要特殊声明,如 global 关键字,初学阶段可以先避免)。
print("\n--- 变量作用域示例 ---")
global_var = "我是一个全局变量"
def scope_test():
local_var = "我是一个局部变量"
print(f"在函数内部,可以看到局部变量: '{local_var}'")
print(f"在函数内部,也可以看到全局变量: '{global_var}'")
# global_var = "尝试在函数内修改全局变量" # 如果没有 global 声明,这会创建一个新的局部变量 global_var
# print(f"在函数内部,修改后的 '全局' 变量: '{global_var}'")
scope_test()
print(f"\n在函数外部,可以看到全局变量: '{global_var}'")
--- 变量作用域示例 --- 在函数内部,可以看到局部变量: '我是一个局部变量' 在函数内部,也可以看到全局变量: '我是一个全局变量' 在函数外部,可以看到全局变量: '我是一个全局变量'
# print(f"在函数外部,不能看到局部变量: {local_var}") # 这会产生 NameError,因为 local_var 只在函数内存在
在我们ctrl跳转到一些函数内部的时候,会发现写法相对我们日常定义的简单函数更加复杂,主要是参数形式比较丰富
参数有以下类型:
可能你还听过关键字参数 (Keyword Arguments)这个说法,但是他并非是一种参数,而是一种传递参数的手段: 调用时通过 参数名=值 的形式指定,可以不按顺序。他可以传位置参数的值,也可以传默认参数的值,也可以传可变参数的值,也可以传关键字参数的值。为了可读性,更推荐对所有参数均采取关键字参数传递。
def describe_pet(animal_type, pet_name):
"""显示宠物的信息。"""
print(f"\n我有一只 {animal_type}.")
print(f"我的 {animal_type} 的名字叫 {pet_name.title()}.")
describe_pet("猫", "咪咪") # 使用关键字参数,顺序不重要
我有一只 猫. 我的 猫 的名字叫 咪咪.
为了可读性,更推荐对所有参数采取关键词参数的写法
假设一个复杂的绘图函数 plot_data(data, x_col, y_col, "blue", "-", True, False, "My Plot", "X-axis", "Y-axis") # 不清晰 使用关键字参数 plot_data(data=my_data, x_column='time', y_column='value', color='blue', linestyle='-', show_grid=True, use_log_scale=False, title="My Awesome Plot", xlabel="Time (s)", ylabel="Value") # 非常清晰
当一个函数有很多参数时,如果只用位置参数,调用者可能需要反复查看函数定义才能确定每个参数的含义。使用关键字参数,每个值的含义都通过其前面的参数名清晰地标示出来。
describe_pet(animal_type="猫", pet_name="咪咪") # 使用关键字参数,顺序不重要
describe_pet(pet_name="旺财", animal_type="狗") # 顺序改变,但结果正确
我有一只 猫. 我的 猫 的名字叫 咪咪. 我有一只 狗. 我的 狗 的名字叫 旺财.
注意点:带默认值的参数必须放在没有默认值的参数之后
def describe_pet_default(pet_name, animal_type="狗"): # animal_type 有默认值
"""显示宠物的信息,动物类型默认为狗。"""
print(f"我有一只 {animal_type}.")
print(f"我的 {animal_type} 的名字叫 {pet_name.title()}.")
describe_pet_default(pet_name="小黑") # animal_type 使用默认值 "狗"
describe_pet_default(pet_name="雪球", animal_type="仓鼠") # 提供 animal_type,覆盖默认值
# 注意:带默认值的参数必须放在没有默认值的参数之后
我有一只 狗. 我的 狗 的名字叫 小黑. 我有一只 仓鼠. 我的 仓鼠 的名字叫 雪球.
*args: 将多余的位置参数收集为一个元组。
当函数被调用时,Python 会先尝试用调用时提供的位置参数去填充函数定义中所有明确定义的、非关键字的形参 (也就是那些普通的,没有 * 或 ** 前缀的参数,包括有默认值的和没有默认值的)。 如果在填充完所有这些明确定义的形参后,调用时还有剩余的位置参数,那么这些“多余的”位置参数就会被收集起来,形成一个元组 (tuple),并赋值给 *args 指定的那个变量(通常就是 args)。 如果调用时提供的位置参数数量正好等于或少于明确定义的形参数量(且满足了所有必需参数),那么 *args 就会是一个空元组 ()。
def make_pizza(size, *toppings):
"""概述要制作的比萨。
*toppings 会将所有额外的位臵参数收集到一个元组中。
"""
print(f"\n制作一个 {size} 寸的比萨,配料如下:")
if toppings: # 只要toppings不为空元组,就会执行
for topping in toppings:
print(f"- {topping}")
else:
print("- 原味 (无额外配料)")
make_pizza(12, "蘑菇")
make_pizza(16, "香肠", "青椒", "洋葱")
make_pizza(9) # toppings 会是空元组
制作一个 12 寸的比萨,配料如下: - 蘑菇 制作一个 16 寸的比萨,配料如下: - 香肠 - 青椒 - 洋葱 制作一个 9 寸的比萨,配料如下: - 原味 (无额外配料)
**kwargs: 将多余的关键字参数收集为一个字典。
当函数被调用时,Python 会先处理完所有的位置参数(包括填充明确定义的形参和收集到 *args 中)。 然后,Python 会看调用时提供的关键字参数 (形如 name=value)。它会尝试用这些关键字参数去填充函数定义中所有与关键字同名的、明确定义的形参(这些形参可能之前没有被位置参数填充)。
如果在填充完所有能通过名字匹配上的明确定义的形参后,调用时还有剩余的关键字参数(即这些关键字参数的名字在函数定义中没有对应的明确形参名),那么这些“多余的”关键字参数就会被收集起来,形成一个字典 (dictionary),并赋值给 **kwargs 指定的那个变量(通常就是 kwargs)。
如果调用时提供的所有关键字参数都能在函数定义中找到对应的明确形参名,那么 **kwargs 就会是一个空字典 {}。
def build_profile(first_name, last_name, **user_info):
"""创建一个字典,其中包含我们知道的有关用户的一切。
**user_info 会将所有额外的关键字参数收集到一个字典中。
"""
profile = {}
profile['first_name'] = first_name
profile['last_name'] = last_name
for key, value in user_info.items():
profile[key] = value
return profile
user_profile = build_profile('爱因斯坦', '阿尔伯特',
location='普林斯顿',
field='物理学',
hobby='小提琴')
print(f"\n用户信息: {user_profile}")
# 输出: {'first_name': '爱因斯坦', 'last_name': '阿尔伯特', 'location': '普林斯顿', 'field': '物理学', 'hobby': '小提琴'}
用户信息: {'first_name': '爱因斯坦', 'last_name': '阿尔伯特', 'location': '普林斯顿', 'field': '物理学', 'hobby': '小提琴'}
*args 和 **kwargs 的核心目的是让函数能够接收不定数量的参数,并以元组和字典的形式在函数内部进行处理。
也就是说 当位置参数用完了 就自动变成*args,当关键词参数用完了 就自动变成**kwarges
# 同时出现 *args 和 **kwargs,注意参数的传入顺序
def process_data(id_num, name, *tags, status="pending", **details): # 注意,这里的status 是仅关键字参数,必须通过关键词传值
print(f"ID: {id_num}")
print(f"Name: {name}")
print(f"Tags (*args): {tags}")
print(f"Status: {status}") # status 是一个有默认值的普通关键字参数
print(f"Details (**kwargs): {details}")
print("-" * 20)
# 调用1:
process_data(101, "Alice", "vip", "new_user", location="USA", age=30)
# ID: 101
# Name: Alice
# Tags (*args): ('vip', 'new_user') <-- "vip", "new_user" 是多余的位置参数,被 *tags 收集
# Status: pending <-- status 使用默认值,因为调用中没有 status=...
# Details (**kwargs): {'location': 'USA', 'age': 30} <-- location 和 age 是多余的关键字参数,被 **details 收集
# --------------------
# 调用2:
process_data(102, "Bob", status="active", department="Sales")
# ID: 102
# Name: Bob
# Tags (*args): () <-- 没有多余的位置参数
# Status: active <-- status 被关键字参数 'active' 覆盖
# Details (**kwargs): {'department': 'Sales'} <-- department 是多余的关键字参数
# --------------------
# 调用3:
process_data(103, "Charlie", "admin") # 'admin' 会被 *tags 捕获
# ID: 103
# Name: Charlie
# Tags (*args): ('admin',)
# Status: pending
# Details (**kwargs): {}
# --------------------
# 调用4: (演示关键字参数也可以用于定义中的位置参数)
process_data(name="David", id_num=104, profession="Engineer")
# ID: 104
# Name: David
# Tags (*args): ()
# Status: pending
# Details (**kwargs): {'profession': 'Engineer'}
# --------------------
ID: 101 Name: Alice Tags (*args): ('vip', 'new_user') Status: pending Details (**kwargs): {'location': 'USA', 'age': 30} -------------------- ID: 102 Name: Bob Tags (*args): () Status: active Details (**kwargs): {'department': 'Sales'} -------------------- ID: 103 Name: Charlie Tags (*args): ('admin',) Status: pending Details (**kwargs): {} -------------------- ID: 104 Name: David Tags (*args): () Status: pending Details (**kwargs): {'profession': 'Engineer'} --------------------
作业:
题目1:计算圆的面积
题目2:计算矩形的面积
题目3:计算任意数量数字的平均值
题目4:打印用户信息
题目5:格式化几何图形描述
1
import math
def calculate_circle_area(radius):
s=math.pi*radius*radius
return s
print(f"半径为5的面积为:{calculate_circle_area(5)}")
print(f"半径为0的面积为:{calculate_circle_area(0)}")
print(f"半径为-1的面积为:{calculate_circle_area(-1)}")
半径为5的面积为:78.53981633974483
半径为0的面积为:0.0
半径为-1的面积为:3.141592653589793
2
def calculate_rectangle_area(length,width):
s=length*width
if length<0 or width<0:
return 0
else:
return s
print(f"长为5,宽为3的面积为:{calculate_rectangle_area(5,3)}")
print(f"长为-1,宽为3的面积为:{calculate_rectangle_area(-1,3)}")
长为5,宽为3的面积为:15
长为-1,宽为3的面积为:0
3
def calculate_average(*args):
if not args:
return 0
total = sum(args)
return total / len(args)
# 示例用法
print(calculate_average(1, 2, 3)) # 输出:2.0
print(calculate_average(10, 20, 30, 40)) # 输出:25.0
print(calculate_average()) # 输出:0
2.0
25.0
0
4
def print_user_info(user_id, **kwargs):
print(f"用户ID: {user_id}")
for key, value in kwargs.items():
print(f"{key}: {value}")
print_user_info(1, name="Alice", age=30, email="[email protected]")
用户ID: 1
name: Alice
age: 30
email: [email protected]
5
def describe_shape(shape_name, color="black", **kwargs):
dimensions = []
for key, value in kwargs.items():
dimensions.append(f"{key}={value}")
if dimensions:
dimensions_str = ", ".join(dimensions)
return f"A {color} {shape_name} with dimensions: {dimensions_str}"
else:
return f"A {color} {shape_name} with no specific dimensions."
print(describe_shape("circle", color="red", radius=5))
print(describe_shape("rectangle", color="blue", width=4, height=6))
A red circle with dimensions: radius=5
A blue rectangle with dimensions: width=4, height=6
@浙大疏锦行