zoukankan      html  css  js  c++  java
  • 20、Python之函数参数的使用

    一、形参与实参介绍

    函数的参数分为形式参数和实际参数,简称形参和实参:

    1.1 形参

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

    def func(x, y):  # x=1,y=2
        print(x, y)
    

    1.2 实参

    实参:在调用函数阶段传入的值,称之为实际参数。值可以是常量、变量、表达式或三者的组合

    实参值的形式:
    #1、实参是常量
    func(1,2)
    
    #2:实参是变量
    a=1
    b=2
    func(a,b)
    
    #3:实参是表达式
    func(int('1'),2)
    

    1.3 形参与实参的关系

    # 1、在调用阶段,实参(变量值)会绑定给形参(变量名)
    # 2、这种绑定关系只能在函数体内使用
    # 3、实参与形参的绑定关系在函数调用时生效,函数调用结束后解除绑定关系
    

    二、形参与实参的具体使用

    2.1 位置参数

    按照从左到右的顺序依次定义的参数称之为位置参数

    2.1.1 位置形参

    在函数定义阶段,按照从左到右的顺序直接定义的"变量名"

    特点:必须被传值,不能多也不能少

    def register(name,age,sex): #定义位置形参:name,age,sex,三者都必须被传值
        print('Name:%s Age:%s Sex:%s' %(name,age,sex))
    register() #TypeError:缺少3个位置参数 
    

    2.1.2 位置实参

    在函数调用阶段,按照从左到右的顺序依次传入的值

    特点:按照顺序与形参一一对应

    2.2 关键字参数

    2.2.1 关键字实参

    在函数调用阶段,按照key=value的形式传入的值

    特点:给指定形参传值,可以完全不参照形参顺序

    2.2.2 与位置参数混合使用

    # 1 位置实参必须放在关键字实参前
    func(y=2,1)  # SyntaxError: positional argument follows keyword argument
    
    # 2 不能能为同一个形参重复传值
    func(1,y=2,x=3)  # TypeError: func() got multiple values for argument 'x'
    func(1,2,x=3,y=4)  # TypeError: func() got multiple values for argument 'x'
    

    2.3 默认参数

    在定义函数阶段,就已被赋值的形参,称之为默认参数

    特点:在定义阶段就已被赋值,意味着在调用阶段可以不用为其赋值

    # def register(name,age,gender='男'):
    #     print(name,age,gender)
    #
    # register('三炮',18)
    # register('二炮',19)
    # register('大炮',19)
    # register('没炮',19,'女')
    

    2.3.1 与位置参数混合使用

    1、位置形参必须在默认形参的左边
    def func(y=2,x):  # SyntaxError: non-default argument follows default argument
        pass
        
    2、默认参数的值是在函数定义阶段被赋值的,准确地说被赋予的是值的内存地址
    示范1:
    m=2
    def func(x,y=m): # y=>2的内存地址
        print(x,y
    m=3333333333333333333  # SyntaxError: invalid syntax
    func(1)
    
    示范2:
    m = [111111, ]
    
    def func(x, y=m): # y=>[111111, ]的内存地址
        print(x, y)
    
    m.append(3333333)
    func(1)  # 1 [111111, 3333333]
    
    3、虽然默认值可以被指定为任意数据类型,但是不推荐使用可变类型
    函数最理想的状态:函数的调用只跟函数本身有关系,不外界代码的影响
    m = [111111, ]
    
    def func(x, y=m):
        print(x, y)
    
    m.append(3333333)
    m.append(444444)
    m.append(5555)
    
    
    func(1)  # 1 [111111, 3333333, 444444, 5555]
    func(2)  # 2 [111111, 3333333, 444444, 5555]
    func
    
    可变类型案例:
    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)
    func(4,5,6)
    
    new_l=[111,222]
    func(1,2,3,new_l)  # [111, 222, 1, 2, 3]
        
    

    2.4 可变长度的参数(*与**的用法)

    可变长度指的是在调用函数时,传入的值(实参)的个数不固定,而实参是用来为形参赋值的,所以对应着,针对溢出的实参必须有对应的形参来接收。

    2.4.1 可变长度的位置参数

    2.4.1.1 *形参名:

    用来接收溢出的位置实参,溢出的位置实参会被*保存成元组的格式然后赋值紧跟其后的形参名

    *后跟的可以是任意名字,但是约定俗成应该是args

    示例1:
    def func(x,y,*z): # z =(3,4,5,6)
        print(x,y,z)
    
    func(1,2,3,4,5,6)  # 1 2 (3, 4, 5, 6)
    
    示例2:
    f my_sum(*args):
        res=0
        for item in args:
            res+=item
        return res
    
    res=my_sum(1,2,3,4,)
    print
    

    2.4.1.2 **可以用在实参中,实参中带,先后的值打散成位置实参

    示例:
    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
    

    2.4.1.3 形参与实参中都带*

    示例:
    def func(x,y,*args): # args=(3,4,5,6)
        print(x,y,args)
    
    func(1,2,[3,4,5,6])  # 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')
    

    2.4.2 可变长度的关键字参数

    2.4.2.1**形参名:

    用来接收溢出的关键字实参,**会将溢出的关键字实参保存成字典格式,然后赋值给紧跟其后的形参名

    **后跟的可以是任意名字,但是约定俗成应该是kwargs

    示例:
    def func(x,y,**kwargs):
        print(x,y,kwargs)
    
    func(1,y=2,a=1,b=2,c=3)  # 1 2 {'a': 1, 'b': 2, 'c': 3}
    

    2.4.2.2 可以用在实参中(* * 后跟的只能是字典),实参中带* * ,先**后的值打散成关键字实参

    def func(x,y,z):
        print(x,y,z)
        
    正确示例:
    func(*{'x':1,'y':2,'z':3}) # func('x','y','z') # x y z
    func(**{'x':1,'y':2,'z':3}) # func(x=1,y=2,z=3) # 1 2 3
    
    错误示例:
    func(**{'x':1,'y':2,}) # func(x=1,y=2)  # TypeError: func() missing 1 required positional argument: 'z'
    func(**{'x':1,'a':2,'z':3}) # func(x=1,a=2,z=3)  # TypeError: func() got an unexpected keyword argument 'a'
    

    2.4.2.3 形参与实参中都带**

    示例:
    def func(x,y,**kwargs):
        print(x,y,kwargs)
    
    func(y=222,x=111,a=333,b=444)  # 111 222 {'a': 333, 'b': 444}
    func(**{'y':222,'x':111,'a':333,'b':4444})  # 111 222 {'a': 333, 'b': 4444}
    

    2.4.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用的
    # 原格式---》汇总-----》打回原形
    

    2.5 命名关键字参数(了解)

    在定义函数时,*后定义的参数,如下所示,称之为命名关键字参数
    特点:
    1、命名关键字实参必须按照key=value的形式为其传值
    def func(x,y,*,a,b): # 其中,a和b称之为命名关键字参数
        print(x,y)
        print(a,b)
    
    # func(1,2,b=222,a=111)
    
    示例
    def func(x,y,*,a=11111,b):
        print(x,y)
        print(a,b)
    
    func(1,2,b=22222)
    

    2.6 组合使用(了解)

    1、形参混用的顺序:位置新参,默认形参,*args,命名关键字形参,**kwargs
    def func(x,y=111,*args,z,**kwargs):
        print(x)
        print(y)
        print(args)
        print(z)
        print(kwargs)
    
    2、实参混用的顺序:
    def func(x,y,z,a,b,c):
        print(x)
        print(y)
        print(z)
        print(a)
        print(b)
        print(c)
    
    # func(111,y=222,*[333,444],**{'b':555,'c':666})
    # func(111,y=222,333,444,b=555,c=666)
    
    # func(111,*[333,444],a=222,**{'b':555,'c':666})
    # func(111,333,444,a=222,b=555,c=66)
    
    # func(111,*[333,444],**{'b':555,'c':666},a=222,)
    func(111,3333,4444,b=555,c=666,a=222)
    
    
    # func(1)
    # func(x=1)
    # func(1,x=1)
    # func(*'hello')
    # func(**{})
    # func(*'hell',**{})
    
  • 相关阅读:
    Android studio快捷键大全 和 eclipse对照(原)
    .net 提取注释生成API文档 帮助文档
    查看443端口被占用无法启动解决办法
    关于正则表达式 C#
    关于 ImageLoader 说的够细了。。。
    什么时候用Application的Context,什么时候用Activity的Context
    关于layoutparam 请铭记。。。。
    java 静态方法上的泛型
    让多个Fragment 切换时不重新实例化
    开源.net 混淆器ConfuserEx介绍
  • 原文地址:https://www.cnblogs.com/zuiyouyingde/p/12520016.html
Copyright © 2011-2022 走看看