参数传进去后到底变不变?

时间:2025-09-07 10:15:01来源:互联网

下面小编就为大家分享一篇参数传进去后到底变不变?,具有很好的参考价值,希望对大家有所帮助。

刚学Python那会儿,我最懵的就是函数参数传递的问题。传进去一个变量,函数里面一顿操作,出来一看,咦?怎么没变?有时候又变了?这Python到底是怎么传递参数的?

就以上问题这次相当于也是做一个总结吧!,保证看完再也不迷糊了!

先说结论:既不是传值也不是传引用!

很多教程会告诉你,Python参数传递是“传对象引用”。这么说没错,但太抽象了。实在点说的话就是:Python里,参数传递相当于把实参的“地址名片”复制一份给形参

关键是后面这点:当你修改参数时,具体行为取决于你操作的是可变对象还是不可变对象

什么是可变和不可变对象?

简单说:

  • 不可变对象:创建后不能修改内容,比如整数、浮点数、字符串、元组
  • 可变对象:创建后可以修改内容,比如列表、字典、集合

来看段代码就明白了:

# 不可变对象示例
a = 10
print(id(a))  # 输出内存地址
a = a + 1
print(id(a))  # 地址变了!

# 可变对象示例
b = [1, 2, 3]
print(id(b))  # 输出内存地址
b.append(4)
print(id(b))  # 地址没变!

image.png

看到没?不可变对象修改后变成了新对象,而可变对象还是在原地修改。

函数参数传递

现在来看函数中的表现,这是最容易踩坑的地方!

情况一:不可变对象作为参数

def change_number(num):
    print(f"函数内修改前: {id(num)}")
    num = 100  # 尝试修改
    print(f"函数内修改后: {id(num)}")
    
x = 10
print(f"函数调用前: {id(x)}")
change_number(x)
print(f"函数调用后: {x}")  # 输出10,没变!
print(f"函数调用后id: {id(x)}")  # 地址也没变

image.png 怎么回事?函数内明明修改了,为什么出来没变?

真相是:num = 100并不是修改了原来的对象,而是创建了一个新对象100,然后让num指向这个新对象。原来的x还是指向原来的对象,当然不变了!

情况二:可变对象作为参数

def change_list(lst):
    print(f"函数内修改前: {id(lst)}")
    lst.append(4)  # 修改列表
    print(f"函数内修改后: {id(lst)}")
    
my_list = [1, 2, 3]
print(f"函数调用前: {id(my_list)}")
change_list(my_list)
print(f"函数调用后: {my_list}")  # 输出[1, 2, 3, 4],变了!
print(f"函数调用后id: {id(my_list)}")  # 地址没变

image.png 这里列表内容确实被修改了,因为列表是可变对象,append操作是在原对象上添加元素,没有创建新对象。

特殊情况:可变对象的重赋值

def reassign_list(lst):
    print(f"函数内重赋值前: {id(lst)}")
    lst = [4, 5, 6]  # 重赋值
    print(f"函数内重赋值后: {id(lst)}")
    
my_list = [1, 2, 3]
print(f"函数调用前: {id(my_list)}")
reassign_list(my_list)
print(f"函数调用后: {my_list}")  # 输出[1, 2, 3],没变!

咦?同样是列表,为什么这里又没变了?

关键区别在于:append是修改原对象,而=重赋值是创建新对象并让形参指向它。这并不影响实参仍然指向原来的对象。

图解参数传递过程

想象一下,每个变量都是一个便利贴,上面写着对象的地址。

当你调用函数时,Python会复制一份便利贴传给函数。现在有两个便利贴都指向同一个对象。

  • 如果对象不可变:当你尝试修改时,Python会创建新对象,然后让你的便利贴指向新地址
  • 如果对象可变:你可以直接通过便利贴找到对象并修改它,所有指向这个对象的便利贴都能看到变化

编程中容易出现的问题

问题一:默认参数用可变对象

def add_item(item, items=[]):  # 坑在这里!
    items.append(item)
    return items

print(add_item(1))  # [1]
print(add_item(2))  # [1, 2] 等等,为什么有1?

因为默认参数在函数定义时就被创建了,每次调用都用的是同一个列表对象!

正确做法:

def add_item(item, items=None):
    if items is None:
        items = []
    items.append(item)
    return items

问题二:以为函数内修改了不可变对象

def update_name(user):
    user["name"] = "李四"  # 这能生效,因为user是可变对象
    
def update_age(age):
    age = 30  # 这不会生效,因为age是不可变对象

person = {"name": "张三"}
update_name(person)  # 正确修改
print(person["name"])  # 李四

age = 25
update_age(age)
print(age)  # 还是25

总结一下

1. Python参数传递是传递对象的“地址名片”

2. 不可变对象(数字、字符串等)在函数内修改不会影响原对象

3. 可变对象(列表、字典等)在函数内修改会影响原对象

4. 但如果是重赋值(=操作),不管是可变还是不可变对象,都不会影响原对象

本站部分内容转载自互联网,如果有网站内容侵犯了您的权益,可直接联系我们删除,感谢支持!