python 定义函数为什么有个长线_python的函数定义中99%的人会遇到的一个坑

列表是一种经常使用的数据类型。在函数的定义中,常常会使用列表作为参数。

比如,要测试一个接口的数据,接口返回的数据格式如下:{  "code": "20000",

"data": ["孙悟空","李白","甄姬"],

"msg": "success",

"status": 0}

要测试的内容是:返回的 data 数据是否跟需求符合。在测试之前,需要对数据进一步处理,比如要增加 “王昭君” 这个元素进去,需要写一个函数:def add_element(data=["孙悟空","李白","甄姬"]):

data.append('王昭君')    return data

print(add_element())

print(add_element())

print(add_element())

在函数定义的时候经常会给参数设置默认值,在这个例子中,将 data 参数设置了默认值,函数定义以后,后面会被频繁的调用,期望值应该是打印如下:["孙悟空","李白","甄姬","王昭君"]

["孙悟空","李白","甄姬","王昭君"]

["孙悟空","李白","甄姬","王昭君"]

实际结果为:["孙悟空","李白","甄姬","王昭君"]

["孙悟空","李白","甄姬","王昭君","王昭君"]

["孙悟空","李白","甄姬","王昭君","王昭君","王昭君"]

原因

当定义函数时,会保存函数中默认参数 data 的值,也就是 ["孙悟空","李白","甄姬"],在每次调用的时候如果传递了新的实参,则使用传递的参数;没有传递,使用定义函数时保存的默认参数。

上面两次调用中,都没有传递新的实参,程序会调用定义函数时保存的默认参数,因为 append() , 在第一次调用以后,默认参数已经由 ["孙悟空","李白","甄姬"] 改变为 ["孙悟空","李白","甄姬","王昭君"],再次执行 append() 之后,就变成了 ["孙悟空","李白","甄姬","王昭君","王昭君"];同理,第三次又改变了。

可以使用 id() 函数来定位问题:def add_element(data=["孙悟空","李白","甄姬"]):

# id 来表示是不是同一个对象

print(id(data))

data.append('王昭君')    return data

print(add_element())

print(add_element())

print(add_element())

打印出来的 id(data) 为同一个对象,也就是默认参数。如果我们改变 第二个 print(add_element())为 print(add_element(["孙悟空","李白","甄姬"])),那么第 2 个 id(data) 就会发生变化,因为它不在是默认值,而是新传进来的实参,实际结果也将变成:2543416926792['孙悟空', '李白', '甄姬', '王昭君']2543418907848["孙悟空","李白","甄姬", '王昭君']2543416926792['孙悟空', '李白', '甄姬', '王昭君', '王昭君']

改进方案如果参数中有列表,尽量不要用它做默认参数

如果使用了列表作为默认参数,函数调用时传入实参,而不是省略

可以在函数体中另外定义一个变量接收默认参数def add_element(data=["孙悟空","李白","甄姬"]):

if data == ["孙悟空","李白","甄姬"]:

data = ["孙悟空","李白","甄姬"]

data.append('王昭君')    return data

作者:王雨泽

原文链接:https://www.cnblogs.com/wagyuze/p/10183262.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值