Python - 可变和不可变对象

简介: Python - 可变和不可变对象

前置知识

  • 在 Python 中,一切皆为对象
  • Python 中不存在值传递,一切传递的都是对象的引用,也可以认为是传址

 

有哪些可变对象,哪些不可变对象?


  • 不可变对象:字符串、元组、数字(int、float)
  • 可变对象:数组、字典、集合

 

不可变对象和可变对象的区别?


  • 可变对象:改变对象内容,对象在内存中的地址不会被改变
  • 不可变对象:改变对象内容,对象在内存中的地址被改变;如果必须存储一个不同的值,则必须创建新的对象

 

不可变对象的应用场景

它们在需要常量哈希值的地方起着重要作用,例如作为字典中的键

 

从内存角度出发说下有什么区别?


不可变对象

image.png

  • Python 中的变量有一个内存空间
  • 具体的数据(对象)也有一个内存空间
  • 而变量保存(指向)的是存储数据(对象)的内存地址,一般也叫对象引用
  • 不可变对象是指对象内容本身不可变
  • 变的是:改变了值,会创建新对象,然后变量改变了对象引用,指向了新对象,旧对象会被垃圾回收

 

可变对象

image.png

变的是:原来对象的内容,不会创建新对象,而变量也还是指向原对象

 

从代码角度看看区别


不可变对象-整型

a = 123
b = a
print(id(a))
print(id(b))
print(a, b)
a += 2
print(id(a))
print(id(b))
print(a, b)
# 输出结果
4473956912
4473956912
123 123
4473956976
4473956912
125 123


  • 从前两次打印可以看到,a、b 变量保存的内存地址是同一个,他们们都保存了 123 的内存地址(123 对象的引用)
  • 预期情况:在 a 做了加法赋值运算之后,既然他们一开始都是指向同一个内存地址,按道理修改 123 后,他们也应该仍然指向同一个内存地址呀,但是并没有!
  • 实际情况:a 指向了新的内存地址,而 b 仍然指向旧的内存地址,所以他们的值也不一样

 

可以看看下面的图

首先,这是一个内存区域

image.png

原理

  • 因为数字(int、float) 是不可变对象,所以不能在 123 的内存地址上直接修改数据
  • 加法赋值,实际上是将原来的 123 复制了一份到新的内存地址,然后再做加法,得到一个新的值 125,最后 a 再指向新的内存地址

 

不可变对象-字符串

a = "test"
b = a
print(id(a))
print(id(b))
print(a, b)
a += "123"
print(id(a))
print(id(b))
print(a, b)
# 输出结果
4455345392
4455345392
test test
4455818288
4455345392
test123 test


不可变对象-元组

a = (1, 2, 3)
b = a
print(id(a))
print(id(b))
print(a, b)
a = a + a
print(id(a))
print(id(b))
print(a, b)
# 输出结果
4455410240
4455410240
(1, 2, 3) (1, 2, 3)
4455359200
4455410240
(1, 2, 3, 1, 2, 3) (1, 2, 3)


可变对象列表

# 列表
a = [1, 2, 3]
b = a
print(id(a))
print(id(b))
print(a, b)
a += [4, 5, 6]
print(a, b)
print(id(a))
print(id(b))
# 输出结果
4327665856
4327665856
[1, 2, 3, 4, 5, 6] [1, 2, 3, 4, 5, 6]
4327665856
4327665856


能看到 a 变量修改值之后,b 的值也随之修改了

 

可以看看下面的图

image.png

  • 因为 list 是可变对象,所以并不会将原来的值复制到新的内存地址再改变,而是直接在原来的内存地址上修改数据
  • 因为 a、b 都是指向原来的内存地址的,所以 a、b 变量保存的内存地址是一致的(对象引用是一致的),当然值也是一样的啦

 

Python 函数的参数传递


这里先提前讲下函数的入门,因为参数传递是个挺重要的点

 

概念

  • 开头有讲到,Python 的一切传递都是对象的引用,函数参数传递也不例外
  • 当传递给函数的是一个变量,实际上传递的是变量保存的对象引用(变量指向的内存地址)
  • 在函数内部修改变量时,会根据变量指向的内存地址,去修改对应的值才对,事实真是如此吗

 

参数传递不可变对象

 
         
# 函数
def test_no_define(age, name):
    age = 123
    name = "poloyy"
    print(age, name)
age = 1
name = "yy"
print(age, name)
test_no_define(age, name)
print(age, name)
# 输出结果
1 yy
123 poloyy
1 yy


参数传递可变对象

# 函数
def test_define(dicts, sets):
    dicts['age'] = 24
    sets.pop()
    print(dicts, sets)
dicts = {"age": 123}
sets = {1, 2}
print(dicts, sets)
test_define(dicts, sets)
print(dicts, sets)
# 输出结果
1 yy
{'age': 123} {1, 2}
{'age': 24} {2}
{'age': 24} {2}


总结

  • 当函数参数传递的变量是不可变对象的时候,函数内改变变量值,函数外的变量不会随之改变
  • 当函数参数传递的变量是可变对象的时候,函数内改变变量值,函数外的变量随之改变
相关文章
|
存储 Java Python
Python不可变对象与可变对象
Python语言是一个以**一切皆对象**的面向对象的动态型语言。Python的对象可以根据其是否可以变化划分为可变对象和不可变对象。
105 0
|
存储 Python
【Python零基础入门篇 · 11】:类型转换和深浅拷贝,可变对象和不可变对象、pass语句
【Python零基础入门篇 · 11】:类型转换和深浅拷贝,可变对象和不可变对象、pass语句
115 0
【Python零基础入门篇 · 11】:类型转换和深浅拷贝,可变对象和不可变对象、pass语句
|
Python
Python的可变对象与不可变对象。
参考:http://thomaschen2011.iteye.com/blog/1441254          Python基础:Python可变对象和不可变对象    http://blog.jobbole.com/49705/                                  Python 中的 is 和 id   Python在heap中分配的对象分成两类:可变对象和不可变对象。
908 0
|
1月前
|
人工智能 数据可视化 数据挖掘
探索Python编程:从基础到高级
在这篇文章中,我们将一起深入探索Python编程的世界。无论你是初学者还是有经验的程序员,都可以从中获得新的知识和技能。我们将从Python的基础语法开始,然后逐步过渡到更复杂的主题,如面向对象编程、异常处理和模块使用。最后,我们将通过一些实际的代码示例,来展示如何应用这些知识解决实际问题。让我们一起开启Python编程的旅程吧!
|
1月前
|
存储 数据采集 人工智能
Python编程入门:从零基础到实战应用
本文是一篇面向初学者的Python编程教程,旨在帮助读者从零开始学习Python编程语言。文章首先介绍了Python的基本概念和特点,然后通过一个简单的例子展示了如何编写Python代码。接下来,文章详细介绍了Python的数据类型、变量、运算符、控制结构、函数等基本语法知识。最后,文章通过一个实战项目——制作一个简单的计算器程序,帮助读者巩固所学知识并提高编程技能。
|
20天前
|
Unix Linux 程序员
[oeasy]python053_学编程为什么从hello_world_开始
视频介绍了“Hello World”程序的由来及其在编程中的重要性。从贝尔实验室诞生的Unix系统和C语言说起,讲述了“Hello World”作为经典示例的起源和流传过程。文章还探讨了C语言对其他编程语言的影响,以及它在系统编程中的地位。最后总结了“Hello World”、print、小括号和双引号等编程概念的来源。
105 80
|
9天前
|
Python
[oeasy]python055_python编程_容易出现的问题_函数名的重新赋值_print_int
本文介绍了Python编程中容易出现的问题,特别是函数名、类名和模块名的重新赋值。通过具体示例展示了将内建函数(如`print`、`int`、`max`)或模块名(如`os`)重新赋值为其他类型后,会导致原有功能失效。例如,将`print`赋值为整数后,无法再用其输出内容;将`int`赋值为整数后,无法再进行类型转换。重新赋值后,这些名称失去了原有的功能,可能导致程序错误。总结指出,已有的函数名、类名和模块名不适合覆盖赋新值,否则会失去原有功能。如果需要使用类似的变量名,建议采用其他命名方式以避免冲突。
30 14