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

刚学Python那会儿,我最懵的就是函数参数传递的问题。传进去一个变量,函数里面一顿操作,出来一看,咦?怎么没变?有时候又变了?这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))  # 地址没变!

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

函数参数传递

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

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

python 复制代码
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)}")  # 地址也没变

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

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

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

python 复制代码
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)}")  # 地址没变

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

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

python 复制代码
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会创建新对象,然后让你的便利贴指向新地址
  • 如果对象可变:你可以直接通过便利贴找到对象并修改它,所有指向这个对象的便利贴都能看到变化

编程中容易出现的问题

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

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

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

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

正确做法:

python 复制代码
def add_item(item, items=None):
    if items is None:
        items = []
    items.append(item)
    return items

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

python 复制代码
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. 但如果是重赋值(=操作),不管是可变还是不可变对象,都不会影响原对象

相关推荐
做运维的阿瑞41 分钟前
Python零基础入门:30分钟掌握核心语法与实战应用
开发语言·后端·python·算法·系统架构
Q_Q196328847544 分钟前
python+spring boot洪涝灾害应急信息管理系统 灾情上报 预警发布 应急资源调度 灾情图表展示系统
开发语言·spring boot·python·django·flask·node.js·php
pop_opo_2 小时前
使用 Python + Pygame 键盘控制无人机(AirSim)
python·无人机·pygame
猿究院-陆昱泽2 小时前
Redis 五大核心数据结构知识点梳理
redis·后端·中间件
yuriy.wang2 小时前
Spring IOC源码篇五 核心方法obtainFreshBeanFactory.doLoadBeanDefinitions
java·后端·spring
程序猿老罗3 小时前
使用Python轻松实现Word到PDF的批量转换
python·pdf·word
咖啡教室4 小时前
程序员应该掌握的网络命令telnet、ping和curl
运维·后端
你的人类朋友5 小时前
Let‘s Encrypt 免费获取 SSL、TLS 证书的原理
后端
老葱头蒸鸡5 小时前
(14)ASP.NET Core2.2 中的日志记录
后端·asp.net
jie*5 小时前
小杰机器学习高级(five)——分类算法的评估标准
人工智能·python·深度学习·神经网络·机器学习·分类·回归