乐趣区

编程语言Python-的参数传递返回值浅拷贝深拷贝

作者:LogM

本文原载于 https://segmentfault.com/u/logm/articles,不允许转载~

1. Python 的参数传递

Python 的参数传递,无法控制引用传递还是值传递。对于不可变对象(数字、字符、元组等)的参数,更类似值传递;对于可变对象(列表、字典等),更类似引用传递。

def fun1(n):
    print(n)   # n 在没修改前,指向的地址和 main 函数中 n 指向的地址相同
    n = 20     # n 在修改后,指向的地址发生改变,相当于新建了一个值为 20 的参数 n

def fun2(l):
    print(l)   # l 在没修改前,指向的地址和 main 函数中 l 指向的地址相同
    l = [5,6,7,8,9]  # l 在修改后,指向的地址和 main 函数中 l 指向的地址仍相同,此时 main 函数中 l 的值也会改变

if __name__=="__main__":
    n = 10
    l = [1,2,3,4,5]

    fun1(n)
    fun2(l)

    print(n)    # n 的值还是 10
    print(l)    # l 的值在 fun2() 中已经被改变 

2. Python 的返回值

Python 的返回值,无法控制引用传递还是值传递。对于不可变对象(数字、字符、元组等)的参数,更类似值传递;对于可变对象(列表、字典等),更类似引用传递。

str1 = 'hi'
l1 = [1,2,3,4,5]

def fun1():
    return str1

def fun2():
    return l1

str2 = fun1()  # 此时,str1 和 str2 的指向地址是相同的
str2 = 'hello'  # 修改后,str2 指向的地址发生改变,相当于新建了一个值为 `hello` 的变量

l2 = fun2()  # 此时,l1 和 l2 的指向地址是相同的
l2 = [6,7,8,9]  # 修改后,l2 指向的地址仍相同,修改 l2 会影响到 l1

3. Python 的浅拷贝和深拷贝

对于不可变对象(数字、字符、元组等),直接赋值的结果和深拷贝一致;对于可变对象(列表、字典等),直接赋值、浅拷贝、深拷贝结果不同。

# 不可变对象(数字、字符、元组等)的例子

a = 100  
b = a
b = 30

print('a:', a)
print('b:', b)

# 结果:# a = 100
# b = 30
# 可变对象(列表、字典等)的例子

import copy

dict1 = {1:1, 'user':'test', 'num':[1, 2, 3]}

dict2 = dict1  # 直接赋值,两个字典指向地址一致
dict3 = dict1.copy()  # 浅拷贝,只拷贝到原对象
dict4 = copy.deepcopy(dict1)  # 深拷贝,除了拷贝原对象,也会拷贝子对象
 
dict1[1] = 11
dict1['user'] = '123'
dict1['num'].remove(1)
 
print('原字典修改后:',dict1)
print('直接赋值:',dict2)
print('浅拷贝:',dict3)
print('深拷贝:',dict4)

# 结果
# 原字典修改后:{1:11,'user': '123', 'num': [2, 3]}
# 直接赋值:{1:11,'user': '123', 'num': [2, 3]}
# 浅拷贝:{1:1,'user': 'test', 'num': [2, 3]}
# 深拷贝:{1:1,'user': 'test', 'num': [1, 2, 3]}
退出移动版