Python函数参数详解

Python函数参数详解

形参与实参

什么是形参


img

  在定义函数阶段定义的参数称之为形式参数,简称形参,相当于变量名。

 

什么是实参


img

  在调用函数阶段传入的值称为实际参数,简称实参。相当于"变量值"的内存地址。(Ps:Python中的所有传值都是传递的内存地址,因此Python中的传值也被称为引用传值。)

 

实参与形参的关系


 

   1.在调用阶段,实参("变量值的内存地址引用")会绑定给形参("变量名")

   2.这种绑定关系只能在函数体内使用

   3.实参与形参的绑定关系在函数调用时生效,函数调用结束后解除绑定关

 

img

 

形参定义方式

位置形参


  在函数定义阶段,按照从左到右的顺序直接定义的"变量名"被称为位置形参,它的特点是必须被传值,多一个少一个也不行

img

 

默认形参


  默认形参的值是在函数定义阶段就被赋予的,准确的说是赋予形参了一个默认的值的内存地址。当调用时可以不对该形参进行传值,如不对该形参进行传值则使用默认的值,反之则使用传入的实参值,需要注意在定义时默认形参必须放在位置形参后面,否则会抛出异常

img

 

实参传值方式

位置传参


  上面一直在使用位置传参的方式,在函数调用阶段,按照从左至右的顺序依次传入值的方式被称为位置传参,特点是按照实参摆放的顺序与形参一一对应进行传值。来看一段错误示范:

img

 

关键字传参


  在函数调用阶段,按照key=value的形式传入值的方式被称为关键字传参,可以打破顺序为形参进行传值。注意:若使用关键字传参需注意必须跟在位置传参的后面。(也是个人比较推崇的一种传值方式)。

img

 

多参函数的形参与实参传值

位置传参与 *


  *是一种语法,args是一种约定俗称。如果定义一个函数时不知道使用者要用位置传参的方式传入多少个实参,可使用*args来进行接收所有多余的被传入的实参,这些多余的被传入实参会被打包成元组类型。args只是形参名,但是按照约定俗称如果要接收多个实参的形参名一般命名为args

# ==== *args 的使用 ====
def func(x,y,*args):
    print("x -->",x)
    print("y -->",y)
    print("args -->",args)
​
func(1,2,3,4,5,"6",[7,8,9]) # 实参 1 和 2 会根据位置传参依次被形参x,y所接收(内存地址)。剩下的所有通过位置传参的实参全部被args所接收。
# ==== 执行结果 ====
"""
x --> 1
y --> 2
args --> (3, 4, 5, '6', [7, 8, 9])
"""

 

关键字传参与 **


  **也是一种语法。kwargs则是后面形参名的约定俗称。如果定义一个函数时不知道使用者要用关键字传参的方式传入多少个实参,可使用**kwargs来进行接收所有多余的被传入的实参,这些多余的被传入实参会被打包成字典类型。

# # ==== **kwargs 的使用 ====
def func(name,age,**kwargs):
    print("name -->",name)
    print("age -->",age)
    print("kwargs -->",kwargs)
​
func("yunya",18,gender="male",height="1.92",body_weight="180Kg")  # 实参yuya,18通过位置传参会被name和age所接受(内存地址),剩下的所有关键字传参的实参全部被kwargs所接收。
# ==== 执行结果 ====
"""
name --> yunya
age --> 18
kwargs --> {'gender': 'male', 'height': '1.92', 'body_weight': '180Kg'}
"""

 

实参中 * 的使用


  *可以用在实参中,实参中带*,先将*后面的值(一般为容器类型,不包含字典类型)拆分为实参后再以位置传值的方式进行传入

# ==== 实参中 * 的使用 ====
def func(x,y,z):
    print("x --->",x)
    print("y --->",y)
    print("z --->",z)
​
func(*["a","b","c"])
​
# ==== 执行结果 ====
"""
x ---> a
y ---> b
z ---> c
"""

 

实参中 ** 的使用


  **也可以用在实参中,实参中带**,先将**后面的值(必须为字典类型)拆分为key=value的形式后再以关键字传值的方式进行传入

# ==== 实参中 ** 的使用 ====
def func(x,y,z):
    print("x --->",x)
    print("y --->",y)
    print("z --->",z)
​
func(**{"x":1,"y":2,"z":3})
​
# ==== 执行结果 ====
"""
x ---> 1
y ---> 2
z ---> 3
"""

 

*args 与 **kwargs的组合使用


  其实关于实参传值的方式就两种,一种是位置传参,一种是关键字传参。所以一个函数如果定义了*args**kwargs后则代表它能够接受足够多的参数。需要注意的是*args必须定义在**kwrags之前,否则会抛出异常。

# ==== *args 与 **kwargs 的组合使用 ====
def wrapper(*args, **kwargs):
    print(args)  # (1, 2, 3)
    print(kwargs)  # {'name': 'Yunya', 'age': 18}
​
​
wrapper(1, 2, 3, name="Yunya", age=18)

 

扩展:命名关键字参数

  我们可以指定某些形参的传值方式必须为关键字传参。那么这些被规定死了传值方式的形参则被称为命名关键字参数,它同位置形参,默认形参是同一级别的。我们先来看一下下面的代码:

img

  我们来看一下默认形参的报错提示是什么:

img

  由于命名关键字参数的应用场景十分少见,Pycharm也会在某些时候进行一些错误的提示:

img

 

扩展:定义形参顺序及传入实参顺序

  注意:以下场景在实际开发中不可能遇到。但是这里还是做一个补充。

img

img

 

扩展:函数嵌套与*和**传参

  这里要提前说一下,下面这个实例是为了给装饰器打基础,装饰器算是Python基础里比较难的一个点。它包含函数嵌套使用与***的多重传参,这里不说那么多。直接上个代码图,看懂了就可以关闭本页面了。

# ==== 函数嵌套与*和**的组合使用 ====
def func(x, y, z):
    print(x, y, z)  # 1 v1 v2
​
​
def wrapper(*args, **kwargs):
    func(*args, **kwargs)  # 实参用 * 和 ** 。拆分开, * 是位置传值,**是关键字传值.
    # args (1) kwargs {"y":"v1","z":"v2"}  拆分成 ---> func(1,y="v1",z="v2") 最终传入样式
​
​
wrapper(1, y="v1", z="v2")

 

扩展:传参类型提示

# ==== 传参类型提示 ====
def func(x:str,y:int,z:list)->"""返回str类型""":
    return "hello,world"print(func.__annotations__)
​
"""
Python3.5之后新增了类型提示功能。在输入参数的时候会显示一定的提示,
注意:这只是一种提示,并不会真正限制你传参时的数据类型。
"""# ==== 执行结果 ====
"""
{'x': <class 'str'>, 'y': <class 'int'>, 'z': <class 'list'>, 'return': '返回str类型'}
"""

 

原文地址:https://www.cnblogs.com/Yunya-Cnblogs/p/12883117.html