python函数参数改变问题

python函数参数改不改变的问题

前几天在做项目的过程中发现了一个问题,向函数中传入一个list,在函数体内将其赋值给list,对list1操作后发现list也发生了变化,啊 ! 出乎意料。查了一下原因,原来python里有可变对象和不可变对象之分。只有传入的是不可变对象时,值才不发生改变,若是可变对象,充当函数参数时要注意了。

不可变对象:Number ,String , Tuple,bool
可变对象: List , Set , Dictionary是可以改变内部的元素

下面总结一下:

先看例子:

def changestr (str):
    str = "inside"
    print "这是function中 , 值为:",str
mystr = "outside"
changestr(mystr)
print "这是函数外边 , 值为:",mystr

输出结果:

这是function中 , 值为: inside
这是函数外边 , 值为: outside

即 传入不可变对象字符串,在函数内对其操作不影响调用结束后字符串的值,即不发生改变。

ps: Number和Tuple结果是一样的,这三种类型只能通过重新赋值来改变对象的值 .

def changestr (str):
    str.append(3)
    print "这是function中 , 值为:",str
mystr = [1,2]
changestr(mystr)
print "这是函数外边 , 值为:",mystr

结果:

这是function中 , 值为: [1, 2, 3]
这是函数外边 , 值为: [1, 2, 3]
  • 对于可变对象,在函数体中的修改 , 对对象本身的值发生了改变 , 在函数之外 , 该列表的内容依然发生了改变 , 这是事先就能猜测到的结果 , 因为python中的参数 , 传入的是变量引用的副本 , 它与变量指向同一个值.

3


def change2(list):
    list = [1,2,3,4]
mylist = ["aa",21]
print(mylist)
change2(mylist)
print(mylist)

输出结果:


['aa', 21]
['aa', 21]

可变对象在函数体中的重新赋值 , 没有对外部变量的值产生影响 , 不过仔细一想 , 却又在情理之中 .

  • 即变量中存储的是引用 , 是指向真正内容的内存地址(当然 ,java中的八大基本数据类型 , 变量名和值都是存储在堆栈中的 ) , 对变量重新赋值 , 相当于修改了变量副本存储的内存地址 , 而这时的变量已经和函数体外的变量不是同一个了, 在函数体之外的变量 , 依旧存储的是原本的内存地址 , 其值自然没有发生改变 .

3

def change2(list):
    list1 =list
    list1.append(34)
mylist = ["aa",21]
print mylist
change2(mylist)
print mylist

输出结果:


['aa', 21]
['aa', 21, 34]
  • 函数体传入的参数 , 为函数体外变量引用的副本 .
  • 在函数体中改变变量指向的堆中的值 , 对函数外变量有效.
  • 在函数体中改变变量的引用 , 对函数外变量无效

要想不改变原list,用copy.deepcopy()


import copy
def change2(list):
    list1=copy.deepcopy(list)
    list1.append(34)
mylist = ["aa",21]
print mylist
change2(mylist)
print mylist

结果:

['aa', 21]
['aa', 21]

要想改变bool型,将其作为返回值

  1. flag不做返回值,函数外不发生改变
flag=True
def change2(list,flag):
    list.append(34)
    flag=False

mylist = ["aa",21]
print mylist
change2(mylist,flag)
print mylist,flag

结果:

['aa', 21]
['aa', 21, 34] True
  1. flag作为返回值,返回值可用,但函数仍然不发生改变
flag=True
def change2(list,flag):
    list.append(34)
    flag=False
    return flag

mylist = ["aa",21]
print mylist
print change2(mylist,flag)
print mylist,flag

输出结果:

['aa', 21]
False
['aa', 21, 34] True

猜你喜欢

转载自www.cnblogs.com/monkey-moon/p/9347505.html
今日推荐