一、函数
使用函数的好处:
1. 同样的功能,可以不需要重复的代码
2. 方便功能修改,代码维护会更方便
3. 代码简洁,逻辑更加清晰
pass => 占位符,什么也不做,让语法结构正确
1. 函数的定义与调用
def 函数名(参数1, 参数2, ...):
def fun01():
pass
# 小案例
def bmi():
weight = float(input("请输入您的体重(kg):"))
height = float(input("请输入您的身高(m):"))
bmi = weight/(height**2)
if bmi > 40:
print(f"BMI = {bmi}, 您的身体质量为3度肥胖。")
elif bmi >= 35:
print(f"BMI = {bmi}, 您的身体质量为2度肥胖。")
elif bmi >= 30:
print(f"BMI = {bmi}, 您的身体质量为1度肥胖。")
elif bmi >= 25:
print(f"BMI = {bmi}, 您的身体质量为超重。")
elif bmi >= 18.5:
print(f"BMI = {bmi}, 您的身体质量为正常。")
else:
print(f"BMI = {bmi}, 您很苗条!")
bmi()
输出:
设计成带参数的函数:
# 形参 => 函数定义时指定的参数
def bmi02(weight, height):
bmi = weight / (height ** 2)
if bmi > 40:
print(f"BMI = {bmi}, 您的身体质量为3度肥胖。")
elif bmi >= 35:
print(f"BMI = {bmi}, 您的身体质量为2度肥胖。")
elif bmi >= 30:
print(f"BMI = {bmi}, 您的身体质量为1度肥胖。")
elif bmi >= 25:
print(f"BMI = {bmi}, 您的身体质量为超重。")
elif bmi >= 18.5:
print(f"BMI = {bmi}, 您的身体质量为正常。")
else:
print(f"BMI = {bmi}, 您很苗条!")
weight = float(input("请输入您的体重(kg):"))
height = float(input("请输入您的身高(m):"))
# 调用 => 实参 => 实际参数,有意义的值
# 形参和实参在内存中不是同一个数据
bmi02(weight, height)
输出:
2. 函数的参数
形参与实参,可同名,但不是同一个对象
形参:函数中的参数名称为形参(函数定义的时候,指定的参数)
实参:提供给函数调用的值成为实参(函数调用的时候,传递的参数)
-
参数分类(按定义,即按形参)
-
必选参数 => 在调用时必须要填
-
默认参数 => 提供默认值,调用时可填可不填
# 让用户调用的时候更方便 def myinfo(name, age, sex="保密"): print(name, age, sex) myinfo('cali', 18, "男") myinfo('cali', 18) # No报错 => age没有 # myinfo(name='cali', sex="男") myinfo(name='cali', age=18)
cali 18 男 cali 18 保密 cali 18 保密 Process finished with exit code 0
-
可变长位置参数
-
可变长关键字参数
-
注意参数定义的先后:
- mysum(必选参数, *args, 默认参数, **kwargs) => 最佳方式
- 不是每项都需要,如果需要的话,按照上述顺序
# 必选参数
def stuinfo(name, age, sex):
print(f"my nameis {name}, my age is {age}, my sex is {sex}")
# 位置参数
# => 按位置依次传递
stuinfo("sc", 21, "f")
# stuinfo("sc", 21) => 传递进去只有两个,函数的必选参数有3个,所以会报错
my nameis sc, my age is 21, my sex is f
Process finished with exit code 0
# 默认参数
def stuinfo(name, age, sex="f"):
print(f"my nameis {name}, my age is {age}, my sex is {sex}")
# 位置参数
stuinfo("sc", 21) # 传递进去两个参数,函数的参数是2个必选参数和1个默认参数,所以不会报错
stuinfo("sc", 21, "m") # 形参的默认参数可以更改,传递3个实参进去
# 关键字参数(有等号的在形参是默认参数形式,在实参就是关键字参数形式)
# => 根据关键字传递
stuinfo("sc", sex="m", age=21)
my nameis sc, my age is 21, my sex is f
my nameis sc, my age is 21, my sex is m
my nameis sc, my age is 21, my sex is m
Process finished with exit code 0
注意参数的调用的顺序
- mysum(位置参数, 关键字参数)
1. 可变长位置参数
def myargs(name, *num):
print(f"{name}: {num}")
print(type(num), *num)
myargs("sc")
myargs("sc1", 1)
myargs("sc2", 1, 2)
sc: ()
<class 'tuple'>
sc1: (1,)
<class 'tuple'> 1
sc2: (1, 2)
<class 'tuple'> 1 2
Process finished with exit code 0
- packing与unpacking
- packing => 实参传入的时候,形参里的星号是打包作用
- 1, 2 => 打包成元组 (1, 2),再赋值给num
- 此时num是元组(1, 2)
- unpacking => 函数中输出时,输出的*是解包作用
- (1, 2) => 解包成1, 2
- print(*num) 即 print(1, 2) => 此时*num输出是1 2
- packing => 实参传入的时候,形参里的星号是打包作用
求和:
- 可以使用遍历求和
def mysum(*args):
# print(args) # 一个元组
sum = 0
for item in args:
sum += item
print(sum)
# 调用 => mysum(1, 2, 3) => 这是位置参数,是有顺序传递的
# => mysum(b=2, a=1) => 这是关键字参数传递 => key:value
mysum(1, 2, 3, 4, 5)
15
Process finished with exit code 0
- 也可以使用sum函数直接求可迭代对象的和
def mysum(*args):
# print(args) # 一个元组
# sum = 0
# for item in args:
# sum += item
# print(sum)
print(f"和为:{sum(args)}")
# 调用 => mysum(1, 2, 3) => 这是位置参数,是有顺序传递的
# => mysum(b=2, a=1) => 这是关键字参数传递 => key:value
mysum(1, 2, 3, 4, 5)
和为:15
Process finished with exit code 0
2. 可变长关键字参数
def myargs2(**num):
print(num)
# print(**num) => print(a=1, b=2) 这样的print是不对的
myargs2(a=1, b=2)
myargs2(a=1)
myargs2(a=1, b=2, c=3)
{'a': 1, 'b': 2}
{'a': 1}
{'a': 1, 'b': 2, 'c': 3}
Process finished with exit code 0
- packing => 将实参传入的a=1, b=2打包成字典{“a”: 1, “b”: 2}
- unpacking => 将字典{“a”: 1, “b”: 2}解包成a=1, b=2
求和:
- 遍历求和
def mysum(**kwargs):
print(kwargs) # 一个字典
sum = 0
for value in kwargs.values():
sum += value
print(sum)
mysum(a=1, b=2, c=3, d=4)
{'a': 1, 'b': 2, 'c': 3, 'd': 4}
10
Process finished with exit code 0
- sum函数求和
def mysum(**kwargs):
print(kwargs) # 一个字典
# sum = 0
# for value in kwargs.values():
# sum += value
# print(sum)
print(sum(kwargs.values()))
mysum(a=1, b=2, c=3, d=4)
{'a': 1, 'b': 2, 'c': 3, 'd': 4}
10
Process finished with exit code 0
# 列表
mylist = [1, 2, 3, 4, 5, 6, 7, 8]
a, *b = mylist
print(f"a={a}\nb={b}")
print("==========")
a, *b, c = mylist
print(f"a={a}\nb={b}\nc={c}")
# 字典
mydict = {'a': 1, 'b': 2, 'c': 3, 'd': 4}
a=1
b=[2, 3, 4, 5, 6, 7, 8]
==========
a=1
b=[2, 3, 4, 5, 6, 7]
c=8
Process finished with exit code 0
def func_tuple(*args):
print(type(args))
for i in args:
print(i)
def func_dict(**dict):
print(type(dict))
print(dict)
if __name__ == '__main__':
t = (1,2,3,'hello')
d = {'a':1,'b':2,'c':3}
func_tuple(*t)
func_dict(**d)
<class 'tuple'>
1
2
3
hello
<class 'dict'>
{'a': 1, 'b': 2, 'c': 3}
Process finished with exit code 0
3. 函数return语句
return语句的作用:
- 用来退出函数
- 函数只要遇到return就会结束执行
- 返回一些数据:
- 如果函数没有return语句,默认返回None;如果return后面不接任何表达式,也是返回None
- return可以有多个,但是只会执行一个return
- return一次可以返回多个值 => 打包成一个元组返回(也可以用多个对象去接收返回值)
# 函数中可以写多个return,但是一次只会执行一个return
def mysum(x):
if x%2:
return "奇数"
else:
return "偶数"
for x in range(4):
print(mysum(x))
偶数
奇数
偶数
奇数
总结
本文介绍了函数的定义与调用,并区分学习了函数的形参与实参,以及掌握函数的分类。我们还要注意在学习函数的return语句时,注意返回值到底是什么。