Python函数的参数列表解析
目录
- 一、函数参数的分类
- 二、形参是对实参的值拷贝
- 三、实参的分类
- 四、形参的分类
- 1、带及不带默认值的位置形参
- 2、带有及不带默认值的关键字形参
- 3、可变长度的形参
- 五、打散机制
- 总结
一、函数参数的分类
函数的参数分为两类:形参 及 实参
形参:形参是在创建过程中声明的参数,如果不给形参传入特定的实参,形参就没有实际的意义
实参:实参是在函数的调用过程中传入的参数,这个参数具有实际意义,具有具体的数据类型
举例:
def fn(n1,n2): # 在此过程中定义的参数就是形参 print(n1 + n2) fn(10,20) # 在此过程中传入的变量就是实参
二、形参是对实参的值拷贝
形参与实参可以重名,但是其表示的是两个变量
举例:
def fn(num1): print(num1) num1 = 20 print(num1) num1 = 10 fn(num1) print(num1)
对于不可变类型:
参考上例,当 num1 的值在函数的内部进行重新赋值时,在外部进行重新打印,打印的是原来的值,原因是在参数内部进行了形参的重指向,实参的值没有发生改变。
对于可变类型:
参考上例,num1 变为可变类型,当 num1 的值在内部使用append()方法进行添加元素时,在函数外打印 num1 ,打印的是添加后的列表,因为没有进行重指向,实参及形参指向的是同一个id
三、实参的分类
实参可以分为 位置实参 及 关键字实参
位置实参 —> 根据位置进行参数的传递
举例:
#Python学习交流:711312441 def fn(a,b,c): print(a) print(b) print(c) fn(1,2,3)
像上方这样传入参数时,实参的传入是按照参数的位置进行传递的,如果改变实参的位置,那么传入参数时就会改变传给形参的值,这样传入参数时的参数就叫做位置实参
fn(a=1,b=2,c=3)
像上方函数的调用这样,如果在调用函数时指名道姓的传入参数,那么不论实参的位置如何变化,形参的得到的值都不会发生变化,这种形式的实参就叫做关键字实参。
注:
1、位置实参是一定按照位置顺序进行参数的传递的,如果传递实参时,实参的顺序发生变化,那么形参的到的值也会发生变化。
2、关键字实参采用的是指明道姓的赋值,所以实参进行位置变化时不会改变形参接收的值
3、在既有关键字实参,又有位置实参时,肯定是先进行位置参数的传值,再进行关键字实参的传值(位置参数在前,关键字参数在后)
四、形参的分类
形参的分类可以分为六种:
- 带有默认值的位置形参
- 不带默认值的位置形参
- 带有默认值的关键字形参
- 不带默认值的关键字形参
- 可变长度的位置形参
- 可变长度的关键字形参
实例:
def fn(a,b,c=10,*,d,e=20,**kwargs)
1、带及不带默认值的位置形参
实例中,在 * 号之前统称为位置形参,位置形参可以由位置实参及关键字实参进行传值。
不带默认值的位置形参:
- 不带有默认值的位置形参必须要实参进行传值,如果实参不传值就会报错
带有默认值的位置形参:
- 带有默认值的关键字位置实参可以用实参进行传值,也可以不使用实参进行传值,不使用实参进行传值时,使用默认值作为其值
2、带有及不带默认值的关键字形参
实例中 * 以后的参数都是关键字形参,在使用关键字形参时,必须使用关键字实参进行传值,不使用会报错
不带默认值的关键字形参:
- 不带有默认值的关键字形参必须使用关键字实参进行传值,如果不传参数就会报错
带有默认值的关键字形参:
- 带有默认值的关键字形参可以用关键字实参进行传值,也可以不传值,不传值时使用默认值作为其值。
3、可变长度的形参
可变长度位置形参:
- 其位置在位置形参及关键字形参之间,当传入的位置实参数量多于位置形参时,多出部分的位置形参会全部存储在可变长度的位置形参中
- 可变长度的关键字形参就是例子中的 * ,只是给这个形参添加了一个名字,这个名字可以自定义,但是约定俗成的命名为 *args
可变长度的关键字形参:
- 可变长度的关键字形参位于参数列表的末尾,当传入的关键字实参多与关键字形参时,多出的部分就会被储存在可变长度的关键字形参中,以元组进行存储
- 可变长度的关键字形参使用 ** 进行表示,此时也需要给关键字形参添加一个名字,这个名字可以自定义,但是约定俗成的称为 **kwargs,以字典进行存储
形参声明时,其声明顺序是有规定的,顺序为:
无默认值位置形参 --> 有关键字位置形参 --> 可变长度的位置形参 --> 无默认值的关键字形参 --> 有默认值的关键字形参 --> 可变长度的关键字形参
常用的参数列表类型:
-- (*args,**kwargs) -- (a,*args) -- (a,b=10,**kwargs) -- (a,*,b,**kwargs) -- (a,*args,b,**kwargs)
在使用形参时要按照位置进行传参,即使是有关键字也要尽量按照位置进行传参,这样更加清晰
五、打散机制
在使用可变长形参时,当传入的值多余需要的值时会把多余的参数放进可变长位置形参 及 可变长关键字形参,但是如果直接传入一个元组及字典中如何直接将其变为元组及字典呢?
此时需要使用 * 及 ** 的打散机制
*单列容器 会打散单列容器
**双列容器 会打散双列容器
print(*(a,b,c)) ---> (a b c) print(**{a:1,b:2}) ---> {a:1,b:2}
附:
在进行字母的对比时,是根据字母的ASCII码表对应的数值进行比较的,会遍历比较
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。