一、形参与实参介绍
形参:在定义函数阶段定义的参数称之为形式参数,简称形参,形参相当于变量名
def func(x,y): #x=1,y=2
print(x,y)
实参:在调用函数阶段传入的值称之为实际参数,简称实参,实参相当于变量值
func(1,2) #在调用阶段函数中形参会申请内存空间
形参与实参的关系:
1.在调用阶段,实参(变量值)会绑定(变量值的内存地址)给形参(变量名)
2.这种绑定关系只能在函数体内使用
3.实参与形参的绑定关系在函数调用时生效,函数调用结束后解除绑定关系
形参是传入的值,但值可以是以下几种:
形式一:
func(1,2)
形式二
a=1
b=2
func(a,b)
形式三
func(int("1"),2)
func(func(1,2),func(2,3)) #只要给函数传入的参数是一个值就行
二、位置参数
位置参数:按照从左到右的顺序依次定义的参数称之为位置参数
1.在函数定义阶段,按照从左到右的顺序直接定义的"变量名"
特点:必须被传值,多一个不行少一个也不行
def func(x,y):
print(x,y)
位置实参
在函数调用阶段,按照从左到右的顺序依次传入的值
特点:按照顺序与形参一一对应
func(1,2)
func(y=1,x=2) #位置实参不需要按照形参的位置传值
三、关键字参数
关键字实参:在函数调用阶段,按照key=values的形式传入的值
特点:指名道姓给某个形参传值,可以完全不参照顺序
def func(x,y):
print(x,y)
混合使用,强调
1.位置实参必须放在关键字实参前
func(1,y=2)
2.不能为同一个形参重复传值
四、默认参数
默认参数:在定义阶段,就已经被赋值的形参,称之为默认参数
特点:在定义阶段就已经被赋值,意味着在调用阶段可以不用为其赋值
def func(x,y=1):
print(x,y)
func(x=1)
func(x=1,y=111)
位置形参与默认形参混用,强调:
1.位置形参必须在默认形参的左边
2.默认参数的值是在函数定义阶段被赋值的,准确的说被赋予的是值的内存地址
3.虽然默认值可以被指定为任意数据类型,但是不推荐使用可变类型
def func(x,y=0):
pass
示例1(不可变类型)
m=1 #变量值1的内存地址被m引用
print("---->",id(m),m)
#----> 10914496 1
m=3333 #重新分配一块内存空间将变量值3333的内存地址重新被m引用,所以内存地址发生变化了
print("---->",id(m),m)
#-----> 140577286953424 3333
def func(x,y=m): #y=3333-->变量值3333的内存地址被m引用,y等于的是值3333的内存地址 #默认参数的值y=3333是在函数定义阶段被赋值的,准确的说被赋予的是值的内存地址
print(x,y)
m=2222 #重新分配一块内存空间将变量值2222的内存地址重新被m引用,所以内存地址发生变化了
print("++++++++>",id(m),m)
#++++++++> 140577286952880 2222
func(1)
#1,3333
示例2(可变类型)
m=[1111,]
print("----------->",id(m),m)
#36966336,[1111]
def func(x,y=m): #y=[1111,]-->变量值列表内存地址被m引用,y等于列表值的内存地址
print(x,y)
m.append(22222) #这块使用的m是上面m引用的内存地址,所以往列表中添加元素,内存地址是不发生变化的
print("+++++++++>",id(m),m)
#36966336,[1111, 22222]
func(1)
#1 [1111.22222]
函数最理想的状态,函数的调用只跟函数本身有关系,不受外界代码的影响
不可预知函数的输出
m=[1111,]
def func(x,y=m): #y=[1111,]-->变量值列表内存地址被m引用,y等于列表值的内存地址
print(x,y)
#操作
m.append(22222) #这块使用的m是上面m引用的内存地址,所以往列表中添加元素,内存地址是不发生变化的
#调用函数
func(1)
#不可预知函数的输出
将定义可变类型的默认参数改成不可变默认参数
xx=None
yy=None
print(id(xx),id(yy))
def func(x,y,z,l=None):
if l is None: #is:比较左右两个值身份id是否相等
l=[]
l.append(x)
l.append(y)
l.append(z)
print(l)
func(1,2,3)
# [1, 2, 3]