1、形参的介绍
形参:在定义函数阶段定义的参数称之为形式参数,简称形参,相当于变量名。 def func(x, y) 实参:在调用函数阶段传入的值称之为实际参数,简称实参,相当于变量值。 func(1,2)
2、形参和实参的关系
在调用阶段,实参(变量值)会绑定给形参(变量名); 这种绑定关系只能在函数体内使用; 实参与形参的绑定关系在函数调用时生效,函数调用结束后解除绑定关系。 实参即是在调用函数时,括号内传入的值,值可以是常量、变量、表达式或三者的组合: 例: 实参是常量 res=my_min(1,2) 实参是变量 a=1 b=2 res=my_min(a,b) 实参是表达式 res=my_min(10*2,10*my_min(3,4)) 实参可以是常量、变量、表达式的任意组合 a=2 my_min(1,a,10*my_min(3,4))
3、形参和实参的具体使用
位置参数:按照从左到右的顺序依次定义的参数称之为位置参数
位置形参:在函数定义阶段,按照从左到右的顺序直接定义的"变量名"
特点:必须被传值,多一个不行少一个也不行
位置实参:在函数调用阶段, 按照从左到有的顺序依次传入的值
特点:按照顺序与形参一一对应
def func(x,y):
print(x,y)
func(1,2,3)错误
func(1,)错误
func(1,2)
关键字参数:
在函数调用阶段,按照key=value的形式传入的值
特点:需指名道姓给某个形参传值,可以完全不参照顺序
例:
def func(x,y):
print(x,y)
func(y=2,x=1)
func(1,2)
混合使用 重点:
位置实参必须放在关键字实参前
func(1,y=2)正确
func(y=2,1)错误
不能能为同一个形参重复传值
错误案例:
func(1,y=2,x=3)
func(1,2,x=3,y=4)
默认参数
默认形参:在定义函数阶段,就已经被赋值的形参,称之为默认参数
特点:在定义阶段就已经被赋值,意味着在调用阶段可以不用为其赋值
def func(x,y=3):
print(x,y)
func(x=1)
func(x=1,y=44444)错误
当函数有多个参数时,需要将值经常改变的参数定义成位置参数,而将值改变较少的参数定义成默认参数。例如编写一个注册学生信息的函数,
如果大多数学生的性别都为男,那完全可以将形参sex定义成默认参数
案例:
def register(name,age,gender='男'):
print(name,age,gender)
register('三炮',18)
register('二炮',19)
register('大炮',19)
register('没炮',19,'女')可以修改
结果:
三炮 18 男
二炮 19 男
大炮 19 男
没炮 19 女
位置形参与默认形参混用:
强调:
位置形参必须在默认形参的左边
def func(y=2,x)错误:
默认参数的值是在函数定义阶段被赋值的,准确地说被赋予的是值的内存地址
示范1:
m=2
def func(x,y=m): # y=>2的内存地址
print(x,y
m=3333333333333333333不能修改
func(1)
结果:1 2
示范2:
m = [111111, ]
def func(x, y=m): # y=>[111111, ]的内存地址
print(x, y)
m.append(3333333) 可以往列表里加值
func(1)
结果:
1 [111111, 3333333]
虽然默认值可以被指定为任意数据类型,但是不推荐使用可变类型
函数最理想的状态:函数的调用只跟函数本身有关系,不外界代码的影响
def func(x,y,z,l=None):
if l is None:
l=[]
l.append(x)
l.append(y)
l.append(z)
print(l)
func(1,2,3) 【1,2,3】
func(4,5,6) 【4,5,6】
new_l=[111,222]
func(1,2,3,new_l) 【111,222,1,2,3,】
可变长度的参数(*与**的用法)
可变长度指的是在调用函数时,传入的值(实参)的个数不固定
而实参是用来为形参赋值的,所以对应着,针对溢出的实参必须有对应的形参来接收
可变长度的位置参数
用来接收溢出的位置实参,溢出的位置实参会被*保存成元组的格式然后赋值紧跟其后的形参名
*后跟的可以是任意名字,但是约定俗成应该是args
案例:
def func(x,y,*z): # z =(3,4,5,6)
print(x,y,z)
func(1,2,3,4,5,6)
用来求和
def my_sum(*args):
res=0
for item in args:
res+=item
return res
res=my_sum(1,2,3,4,)
print(res)
*可以用在实参中,实参中带*,先*后的值打散成位置实参
def func(x,y,z):
print(x,y,z)
# func(*[11,22,33]) # func(11,22,33)
# func(*[11,22]) # func(11,22)
l=[11,22,33]
func(*l) #11 22 33
形参与实参中都带*
def func(x,y,*args): # args=(3,4,5,6)
print(x,y,args)
func(1,2,[3,4,5,6])
func(1,2,*[3,4,5,6]) # func(1,2,3,4,5,6)
func(*'hello') # func('h','e','l','l','o')
**形参名:用来接收溢出的关键字实参,**会将溢出的关键字实参保存成字典格式,然后赋值给紧跟其后的形参名
**后跟的可以是任意名字,但是约定俗成应该是kwargs
>>> def foo(x,**kwargs): #在最后一个参数kwargs前加**
... print(x)
... print(kwargs)
...
>>> foo(y=2,x=1,z=3) #溢出的关键字实参y=2,z=3都被**接收,以字典的形式保存下来,赋值给kwargs
1
{'z': 3, 'y': 2}
**可以用在实参中(**后跟的只能是字典),实参中带**,先**后的值打散成关键字实参
def func(x,y,z):
print(x,y,z)
func(*{'x':1,'y':2,'z':3}) # func('x','y','z')
func(**{'x':1,'y':2,'z':3}) # func(x=1,y=2,z=3)
错误
func(**{'x':1,'y':2,}) # func(x=1,y=2)
func(**{'x':1,'a':2,'z':3}) # func(x=1,a=2,z=3)
形参与实参中都带** 实参除了和形参中对应的,其他都绑定给kwargs转成列表
def func(x,y,**kwargs):
print(x,y,kwargs)
func(y=222,x=111,a=333,b=444)
func(**{'y':222,'x':111,'a':333,'b':4444})
结果:
111 222 {'a': 333, 'b': 444}
111 222 {'a': 333, 'b': 4444}
混用*与**:*args必须在**kwargs之前
def func(x,*args,**kwargs):
print(args)
print(kwargs)
func(1,2,3,4,5,6,7,8,x=1,y=2,z=3)
def index(x,y,z):
print('index=>>> ',x,y,z)
def wrapper(*args,**kwargs): #args=(1,) kwargs={'z':3,'y':2}
index(*args,**kwargs)
# index(*(1,),**{'z':3,'y':2})
# index(1,z=3,y=2)
wrapper(1,z=3,y=2) # 为wrapper传递的参数是给index用的
来源:https://www.cnblogs.com/200024mc/p/12520706.html