初学python,调用固定参,不固定参函数遇到的问题(持续更新)

函数调用时的参数传递方式:(调用函数) 传递方式有以下四种: 位置传参 序列传参 关键字传参 字典关键字传参
---------------------------------------------------yu---

(1)位置传参:
实际参数(实参)的对应关系与形式参数(形参)对应关系是按位置依次对应的
实参的个数和形参的个数要严格一直
实例:

def myfun1(a, b, c):   # 形参
            print(a)
            print(b)
            print(c)
      # 调用
      myfun1(1, 2, 3)   # 实参

(2)序列传参
序列传参是指在函数调用过程中,用*将序列拆解后按位置进行传递的方式
序列传参时,序列拆解的位置将与形参一一对应
序列的位置信息对应相应的参数
实例:

# 假设已经有序列可用
      L1 = [11,22,33]
      T1 = (4.4,5.5,6.6)
      S1 = "ABC"
      myfun1(*L1)
      myfun1(*T1)
      myfun1(*S1)

(3)关键字传参:
关键字传参是指传参时,按形参的名称给形参赋值
实参和形参按形参名进行匹配(可以不按位置顺序进行匹配)
实例:

# 关键字传参
      myfun1(c = 33, b = 22, a = 11)

(4)字典关键字传参:
是指实参为字典,将字典用**拆解后进行关键字传参
实例:

d1 = {"c":33, "b":22, "a":11}
      myfun1(**d1)

说明:字典的键名和形参名必须一
字典的键名必须为字符串,并符合关键字的命名
字典的键名要在形参中存在

(5)函数的综合
传参方式,在能确定每个形参否能唯一匹配一个实参的情况也可以任意传参
传参时,位置传参和序列传参要先于关键字传参和字典关键子传参
实例: myfun1(100, *[200, 300])

myfun1(*[10,20],30)
         myfun1(*[10], 20, *[30])
         myfun1(100, **{"c":300, "b":200})
         myfun1(**{"c":300, "b":200}, a = 100)
         myfun1(b = 200, c = 300, 100)   # 错的

注意传参时:
1、不可变类型的数据作为函数参数传入时,函数内部不会改变变量的原数据的值
2、可变类型的数据作为函数参数传递时,函数内部可以改变原数据,多用与返回更多函数执行结果
实例:写一个函数,在函数内输入一些整数,

(1)把所有奇数放入全局列表odds中
  (2)把所有的偶数放入全局的列表evens中
odds = []
evens = []
def input_number(o, e):
    while True:
        n = int(input("请输入整数:"))
        if n < 0:
            break
        # 如果n为奇数则添加列表odds中
        if n % 2 == 1:
            o.append(n)
        else: # 如果n为偶数则添加到列表evens中
            e.append(n)
input_number(odds, evens)
print(odds)   # 此列表将被修改
print(evens)    # 此列表也将被修改

2、函数参数的定义规则(创建函数)
函数的缺省参数:
语法:def 函数名(形参名1 = 默认实参1, 形参名2 = 默认实参2, ....):

语句块

实例:以下示意函数的缺省参数
def info(name, age=1, address="未填写"):

print(name, "住在:", address, "今年:", age, "岁")

info("庄AC", 23, "北京")
info("庄AC", 23)
说明:(1)缺省参数必须自右至左依次存在,如果一个参数有缺省参数,则其右侧的所有参数都必须有缺省参数

如:def test_fn(a, b = 10, c):    # 是错的
     (2)缺省参数可以有0个或多个,甚至全部都有缺省参数

3、函数的参数定义方式:

位置形参
 星号元组形参
 命名关键字形参
 双星号字典形参

(1)位置形参语法:
def 函数名(形参名1, 形参名2, 形参名3,......):

语句块

(2)星号元组形参语法:
def 函数名(*元组形参名):

语句块

作用:收集多余的位置传参
实例:

def funs(*args):
       # args绑定一个元组,此元组内的元素是为了剩余的位置传参
       print("施参个数是:", len(args))
       print("args绑定", args)
   funs()  # 以无参调用
   funs(1, 2, 3, 4)
   funs("ABC", 1, 2.3, 4+5J)

(3)命名关键字形参语法:
def 函数名(*, 命名关键字形参):
语句块

def 函数名(*args, 命名关键字形参):
语句块
作用:所有的命名关键字形参,必须用关键字传参或字典关键字传参进行参数传递
实例:

def fn(*, a, b):
       print("a的值为:", a)
       print("b的值为:", b)
   # fn(1, 2)  # 错的,a和b 必须是用关键字传参
   fn(a = 1, b = 2)  #必须这么传
   fn(b = 0, a = 20)
   fn(**{"b":200, "a":100})

def f2(*args, a, b):
       print("args=", args)
       print("a=", a)
       print("b=", b)
   f2(1, 2, 3, 4, a=100, b=200)
   f2(a=100, b=200)

(4)双星号字典形参语法:
def 函数名(**字典形参名):
语句块

作用:收集多余的关键字传参(例如:dict()),通常只有一个
实例:

def func(**kwagrs):  # kwagrs绑定一个字典
       print("参数个数:", len(kwagrs))
       print("kwagrs的值为:", kwagrs)
   func(name="zhuang", age=35, address="dgas")
   func()

(5)函数参数自左至右的顺序为:
位置形参, 星号元组形参, 命名关键字形参, 双星号字典形参
实例:

def fn(a, b, *args, c, **kwargs):
      pass
fn(100,200,300,400,c=5.5,d=6.6,e=7.7)
fn(10,*"AB",20,*{"a":100, "b":200, "c":300})

# ----------以下函数可以接受任意的位置传参和关键字传参------------
def fn(args, *kwargs):





额外部分一:

python函数参数支持带星号*的非固定个数参数,如:

def getInfos(msg,*users):
    for u in users:
        print(u)

在调用时,可以往users传入一个元组或列表,如果传入的列表或元组前带星号,刚传入后会自动拆包。具体的不同用法有不同的效果,如下:

1、不定形参传入元组。

getInfos('msg',('aa','bb') )

输出:

('aa', 'bb')

2、不定形参传入带星元组。

getInfos('msg',*('aa','bb') )

输出:

aa
bb

3、不定形参传入列表。

getInfos('msg',['aa','bb'] )

输出:

['aa', 'bb']


光看结果可能像是列表!!!!!!!!!!!!!但其实返回的就是一个元组
可通过以下代码进行验证:

#!/usr/bin/python3
def getInfos(msg,*users):
    users[0]="hh"
    for u in users:
        print(u)
a = ['aa','bb']
getInfos('msg',a )

结果:

Traceback (most recent call last):
  File "/tmp/653879115/main.py", line 7, in 
    getInfos('msg',a )
  File "/tmp/653879115/main.py", line 3, in getInfos
    users[0]="hh"
TypeError: 'tuple' object does not support item assignment

exit status 1

这表明是一个元组。

4、不定形参传入带星列表。

getInfos('msg',*['aa','bb'] )

输出 :

aa
bb

5、直接传入多个值。

getInfos('msg', 'aa','bb')

输出:

aa

额外部分二:

看如下例子:

def singalStar(common, *rest):
    print("Common args: ", common)
    print("Rest args: ", rest)

第一种方式,星号(*)参数不传参:

singalStar("hello")
     
    #Common args:  hello
    #Rest args:  ()

第二种方式,传多个值(个数大于或等于函数定义时的参数个数):

singalStar("hello", "world", 000)
     
    #Common args:  hello
    #Rest args:  ('world', 0)

不难看出,上述方式中,星号参数把接收的参数合并为一个元组。

第三种方式,竟然星号参数把接收的参数作为元组,那么我们直接传元组类型的值:

singalStar("hello", ("world", 000))
     
    #Common args:  hello
    #Rest args:  (('world', 0),)

没错,你没看错,传递的元组值作为了星号参数的元组中的一个元素。

第四种方式,但是有时候我们想把元组值就作为星号参数的参数值,那么该怎么办呢?好办,在元组值前加上“”即可,不过此时,就不能在加了“”的元组后,追加任何值了。

singalStar("hello", *("world", 000))

singalStar("hello", *("world", 000), "123") #error

Common args: hello

Rest args: ('world', 0)

3.带两个星号(*)的函数参数

带两个星号(*)的函数定义如下:

def doubleStar(common, **double):
        print("Common args: ", common)
        print("Double args: ", double)

第一种方式,星号(*)参数不传值:

doubleStar("hello")

Common args: hello

Double args: {}

第二种方式,传多个参数(个数大于或等于函数定义时的参数个数)。但是,这和单星号参数传值方式肯定不一样,否则,不就乱套了吗。

doubleStar("hello", "Test", 24) #error
doubleStar("hello", x = "Test", y = 24)

Common args: hello

Double args: {'y': 24, 'x': 'Test'}

不难发现,此时必须采用第一节的默认值传参的“args = value”的方式。同时,“=”前的字段成了字典的键,“=”后的字段成了字典的值。即,双星号参数接收的参数作为字典。

第三种方式,有时候我们想把字典值就作为星号参数的参数值,那么该怎么办呢?同单星号参数,在字典值前加上“**”,同时其后不能添加任何值。

#doubleStar("hello", **{"name": "Test", "age": 24}, {"name": "Test2", "age": 24})    #error
#doubleStar("hello", {"name": "Test", "age": 24})    #error
doubleStar("hello", **{"name": "Test", "age": 24})
 
#Common args:  hello
#Double args:  {'name': 'Test', 'age': 24}

在有些情况下,单星号函数参数和双星号函数参数是一起使用的,定义如下:

def singalAndDoubleStar(common, *single, **double):
        print("Common args: ", common)
        print("Single args: ", single)
        print("Double args: ", double)

4.总结

默认值函数参数。这种函数定义时,第一个有默认值的参数后的每一个参数都必须提供默认值。传参时,可以直接传参,也可以以“默认值参数名=value”的形式传参。
单星号函数参数。单星号函数参数接收的参数组成一个元组。
双星号函数参数。双星号函数参数接收的参数组成一个字典。

完整的代码如下:

def singalStar(common, *rest):
    print("Common args: ", common)
    print("Rest args: ", rest)
    
def doubleStar(common, **double):
    print("Common args: ", common)
    print("Double args: ", double)
    
def singalAndDoubleStar(common, *single, **double):
    print("Common args: ", common)
    print("Single args: ", single)
    print("Double args: ", double)
 
def defaultValueArgs(common, defaultStr = "default", defaultNum = 0):
    print("Common args", common)
    print("Default String", defaultStr)
    print("Default Number", defaultNum)   
 
if __name__ == "__main__":
    defaultValueArgs("Test")
    defaultValueArgs("Test", "default", defaultNum = 1)
     
    singalStar("hello")
    singalStar("hello", "world", 000)
    singalStar("hello", ("world", 000))
    singalStar("hello", ("world", 000), {123: 123})
    singalStar("hello", *("world", 000))
#     singalStar("hello", *("world", 000), "123")    #error
    
    doubleStar("hello")
    doubleStar("hello", x = "Test", y = 24)
    doubleStar("hello", **{"name": "Test", "age": 24})
#     doubleStar("hello", {"name": "Test", "age": 24})    #error
 
    singalAndDoubleStar("hello")
    singalAndDoubleStar("hello", "world", 000)
    singalAndDoubleStar("hello", "world", 000, {"name": "Test", "age": 24})
    singalAndDoubleStar("hello", "world", 000, **{"name": "Test", "age": 24})
    singalAndDoubleStar("hello", ("world", 000), {"name": "Test", "age": 24})
#     singalAndDoubleStar("hello", *("world", 000), {"name": "Test", "age": 24})      #error
    singalAndDoubleStar("hello", *("world", 000), **{"name": "Test", "age": 24})

相关推荐