今天的我更强了 发表于 2021-7-16 16:16:03

关于复制问题

def func(ls=[]):
    ls.append(1)
    return ls
a=func()
b=func()
print(a,b)
求大佬解释一下为什么答案是

灰晨 发表于 2021-7-16 16:29:07

传入的参数为变量ls,指向的[]在定义函数之前创建,而不是每次执行前重新创建一个新列表对象

青出于蓝 发表于 2021-7-16 16:35:14

def func(ls):
    ls.append(2)
    print(ls)
    return ls
a=func([])
b=func([])
print(a,b)

在函数后面写参数的括号内定义列表,集合,字典都会这样

青出于蓝 发表于 2021-7-16 16:36:10

灰晨 发表于 2021-7-16 16:29
传入的参数为变量ls,指向的[]在定义函数之前创建,而不是每次执行前重新创建一个新列表对象

那为何如果ls是字符就没事勒?

Mzzith 发表于 2021-7-16 16:46:05

a = func()时 调用一次 b = func()时又调用一次 且进行存储print(a,b)时 就已经执行了俩次func()

灰晨 发表于 2021-7-16 17:14:49

青出于蓝 发表于 2021-7-16 16:36
那为何如果ls是字符就没事勒?

像你的代码这样,只是 ls,就代表你这个参数没有默认值
没有默认值,你调用的时候就必须传参
a=func([])
b=func([])
这时 ls 这个变量指向的就是你传入的 []
你在函数里开头打印一下它的iddef func(ls):
    print(id(ls))
    ls.append(2)
    print(ls)
    return ls
a=func([])
b=func([])
print(a,b)

会发现,就算同样是 [] ,它们的储存地址也是不一样的,是不同的两个列表
而有默认值的参数,该默认值在定义函数前就创建了,类似于a = []
def func(ls):
    print(id(ls))
    ls.append(2)
    print(ls)
    return ls
a=func(a)
b=func(a)
print(a,b)

就是说,你参数有默认值,而之后调用参数的时候没有进行传参,则 变量ls默认指向你的默认值的id,如
def func(ls=[]):
    print(id(ls))
    ls.append(2)
    print(ls)
    return ls
a=func()
b=func()
c=func()
print(a,b,c)

a没有进行传参,所以,ls默认指向[],在[]中添加2,成
b进行了传参,则,ls指向传入的参数,在该列表中添加2,成
注,此时实际有两个列表了,一个,另一个
c同样没有进行传参,所以,ls依旧指向默认值的id,而此时该默认值为
列表中的值虽然变了,不过列表还是默认的那个列表,在该列表中添加2,成
即,此时python存储了两个列表,一个是默认的列表,一个是你自己定义的列表,
一个为, 一个为

灰晨 发表于 2021-7-16 17:19:50

建议,默认值最好还是别设定成可变对象
代码很少还好说,代码一多,别人看起来,不,自己之后看都得疯

青出于蓝 发表于 2021-7-16 17:24:45

灰晨 发表于 2021-7-16 17:19
建议,默认值最好还是别设定成可变对象
代码很少还好说,代码一多,别人看起来,不,自己之后看都得疯

这个我懂,主要是按楼主的代码,他是在函数后那个传入参数括号内定义的变量,就出现了问题。但楼主将ls命名为了列表,如果ls是字符串,就不会有问题了。

灰晨 发表于 2021-7-16 17:35:48

本帖最后由 灰晨 于 2021-7-16 17:41 编辑

青出于蓝 发表于 2021-7-16 17:24
这个我懂,主要是按楼主的代码,他是在函数后那个传入参数括号内定义的变量,就出现了问题。但楼主将ls ...

因为字符串是不可变对象啊,比如说
def func(ls='a'):
    print(ls)
    print(id(ls))
    ls = ls + 'b'
    print(ls)
    print(id(ls))
    return ls
a=func()
b=func()
print(a,b)
你看,ls 指向默认值'a','a'的id是没有变,假如'a'的id是1234
那么,调用函数不传参,则,ls——>1234——>a
按楼主的代码,假如 [] 的id为789,
那么,调用函数不传参,则,ls——>789——>[]
但列表是可变对象啊,append(1)后[]就成了,可是列表还是那个列表,它的id没有变的
之后再调用函数不传参,就是,ls——>789——>

ls = ls + 'b'
这里改变了 ls 的指向,ls 现在的指向是 'ab' , 'ab'的id 跟 'a' 的id 肯定不一样了
之后b=func(),此时再次调用还是不传参,ls 依旧是指向默认值,而默认值就是 id为 1234的 'a',而不是'ab'

Kayko 发表于 2021-7-17 07:00:43

{:10_266:}
页: [1]
查看完整版本: 关于复制问题