Python 的可变和不可变对象详情

目录
  • Python 中的可变和不可变对象
    • 一、文字描述可变和不可变对象
      • 1、可变与不可变对象归类
      • 2、可变与可变对象的区别
      • 3、不可变对象的应用场景
    • 二、代码角度区别
      • 1、不可变对象-整型
      • 2、不可变对象-字符串
      • 3、不可变对象-元组
      • 4、可变对象列表
    • 三、Python 函数的参数传递
      • 1、参数传递不可变对象
      • 2、参数传递可变对象

Python 中的可变和不可变对象

一、文字描述可变和不可变对象

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

1、可变与不可变对象归类

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

2、可变与可变对象的区别

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

3、不可变对象的应用场景

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

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

不可变对象:

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

可变对象:

 

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

二、代码角度区别

1、不可变对象-整型

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 仍然指向旧的内存地址,所以他们的值也不一样

可以看看下面的图

首先,这是一个内存区域

原理:

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

2、不可变对象-字符串

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

3、不可变对象-元组

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)

4、可变对象列表

# 列表
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 的值也随之修改了

可以看看下面的图

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

三、Python 函数的参数传递

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

概念:

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

1、参数传递不可变对象

# 函数
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

2、参数传递可变对象

# 函数
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}

总结:

  • 当函数参数传递的变量是不可变对象的时候,函数内改变变量值,函数外的变量不会随之改变
  • 当函数参数传递的变量是可变对象的时候,函数内改变变量值,函数外的变量会随之改变

以上就是Python 的可变和不可变对象详情的详细内容,更多关于Python 的可变和不可变对象的资料请关注我们其它相关文章!

(0)

相关推荐

  • Python面向对象之成员相关知识总结

    1.成员 1.1 变量 实例变量,属于对象,每个对象中各自维护自己的数据. 类变量,属于类,可以被所有对象共享,一般用于给对象提供公共数据(类似于全局变量). class Person(object): country = "中国" def __init__(self, name, age): self.name = name self.age = age def show(self): # message = "{}-{}-{}".format(Person.co

  • Python中可变和不可变对象的深入讲解

    目录 前置知识 有哪些可变对象,哪些不可变对象? 不可变对象和可变对象的区别? 不可变对象的应用场景 从内存角度出发说下有什么区别? 不可变对象 可变对象 从代码角度看看区别 不可变对象-整型 不可变对象-字符串 不可变对象-元组 可变对象列表 Python 函数的参数传递 概念 参数传递不可变对象 参数传递可变对象 总结 前置知识 在 Python 中,一切皆为对象 Python 中不存在值传递,一切传递的都是对象的引用,也可以认为是传址 有哪些可变对象,哪些不可变对象? 不可变对象:字符串.

  • Python中的类对象示例详解

    抽象特点 Python 一切皆对象,基于此概念,对 类 class 有以下特点: 类与实例的属性 类对象创建可选择定义类属性,创建实例对象时,实例属性自动执行类的__init__方法初始化 实例对象自动继承相应的类属性(如果有),但实例属性优先级更高 实例方法,类方法,静态方法的参数 实例方法是一般函数但实例方法需要传入self参数(与一般函数的区别) 类方法和静态方法是通过装饰器实现的函数,类方法需要传入cls参数,静态方法无需传入self参数或者是cls参数(但也能传入参数) 其中self参

  • 简单谈谈Python面向对象的相关知识

    一.私有化 上篇说过封装,既将我们不想让别人看到代码的内容,但是又需要用到的内容,通过类内部调用来实现调用. 说到这里却不得不提一下上篇的: class Person(object): def __init__(self, name, age): self.xxx = name self.xxxx = age 这里面self后面的名字,是可以自己随意命名的,上一篇和后面一样只是为了好记忆罢了 只要你记得住,便是颠倒也是无事 1.1 属性私有化 何为属性私有? 举个例子便是:你的私房钱,你的手机电

  • python面向对象基础之常用魔术方法

    一.类和对象 通俗理解:类就是模板,对象就是通过模板创造出来的物体 类(Class)由3个部分构成: 类的名称: 类名 类的属性: 一组数据 类的方法: 允许对进行操作的方法 (行为) 二.魔法方法 在python中,有一些内置好的特定的方法,方法名是"__xxx__",在进行特定的操作时会自动被调用,这些方法称之为魔法方法.下面介绍几种常见的魔法方法. 1.__init__方法 :初始化一个 类 ,在创建实例对象为其 赋值 时使用. 2.__str__方法:在将对象转换成字符串 st

  • Python面向对象之内置函数相关知识总结

    Python内置函数 1. classmethod.staticmethod.property . 上述三个内置函数在文章(Python进阶--面向对象之成员)的方法和属性中已经详细介绍使用,可以返回浏览,具体地址: 2. callable,是否可在后面加括号执行. 函数 def func(): pass print( callable(func) ) # True 类 class Foo(object): pass print( callable(Foo) ) # True 类中具有__cal

  • python面向对象之类的继承详解

    一.概述 面向对象编程 (OOP) 语言的一个主要功能就是"继承".继承是指这样一种能力:它可以使用现有类的所有功能,并在无需重新编写原来的类的情况下对这些功能进行扩展. 通过继承创建的新类称为"子类"或"派生类",被继承的类称为"基类"."父类"或"超类",继承的过程,就是从一般到特殊的过程.在某些 OOP 语言中,一个子类可以继承多个基类.但是一般情况下,一个子类只能有一个基类,要实

  • Python 可迭代对象 iterable的具体使用

    目录 前置知识 可迭代对象 如何判断一个对象是否是可迭代对象? enumerate 函数 多嵌套列表 总结 前置知识 如果给定一个 list 或 tuple,我们可以通过 for 循环来遍历这个 list 或 tuple,这种遍历我们称为迭代(Iteration) 在 Python 中,迭代是通过 for ... in 来完成的 lists = [1, 2, 3, 4, 5] for i in lists: print(i) 可迭代对象 for 循环不仅可以用在 list 或 tuple 上,还

  • python学习之可迭代对象、迭代器、生成器

    Iterable – 可迭代对象 能够逐一返回其成员项的对象. 可迭代对象的例子包括所有序列类型 (例如 list, str 和 tuple) 以及某些非序列类型例如 dict, 文件对象以及定义了__iter__()方法或是实现了序列语义的__getitem__() 方法的任意自定义类对象. 可迭代对象可用于 for 循环以及许多其他需要一个序列的地方(zip().map() -).当一个可迭代对象作为参数传给内置函数 iter() 时,它会返回该对象的迭代器.这种迭代器适用于对值集合的一次性

  • python获取对象信息的实例详解

    1.获取对象类型,基本类型可以用type()来判断. >>> type(123) <class 'int'> >>> type('str') <class 'str'> >>> type(None) <type(None) 'NoneType'> 2.如果想获得一个对象的所有属性和方法,可以使用dir()函数返回包含字符串的list. >>> dir('ABC') ['__add__', '__cl

  • python面向对象版学生信息管理系统

    本文实例为大家分享了python面向对象版的学生信息管理系统的具体代码,供大家参考,具体内容如下 功能: 1.能循环输入内容 2.展示功能 -1.新增名片 -2.显示全部 -3.查询名片 -0.退出系统 3.让用户输入希望执行的操作 例如输入1,2,3,0 4.新增列表 提示用户输入姓名, 输入电话 输入qq号 输入邮箱 提示新增名片成功. 5.显示全部 将所有的名片打印出来,格式如下: 姓名    电话       QQ               邮箱 zs       110    133

随机推荐