在系统地学习完python后回过头来看python变量,小编又有了不一样的感悟。今天我们再次从更底层的角度来看python变量,从内存地址来分析python变量的行为。看看能不能得到一些更有意义的知识。

一、Python变量

在大多数语言中,为一个值起一个名字时,把这种行为称为“给变量赋值”或“把值存储在变量中”。不过,Python与许多其它计算机语言的有所不同,它并不是把值存储在变量中,而像是把名字“贴”在值的上边(专业一点说法是将名字绑定了对象)。所以,有些Python程序员会说Python没有变量,只有名字,通过名字找到它代表的值。

 Python中的变量,与其它开发语言(如C语言)的不同:

在C语言中,变量类似于一个“容器”,赋给它的值,装在容器中:

定义一个变量 int a = 1;

给变量a重新赋值 a = 2;

把变量a赋值给另外一个变量b ,int b = a;

会重新创建一个变量b(容器),将a中的内容复制粘贴至b中。

在python,变量类似于名字标签“贴”在值上面,通过名字找到它代表的值。

定义一个变量 a = 1

给变量a重新赋值 a = 2

把变量a赋值给另外一个变量b, b = a

创建新的便利贴b,与a同时贴到值上

为了对python中变量的这种情况加深认识,下面适度展开介绍。

1.1 第一点

先说明第一点:变量的实现方式有:引用语义、值语义

python语言中变量的实现方式就是引用语义,在变量里面保存的是值(对象)的引用(值所在处内存空间的地址)。采用这种方式,变量所需的存储空间大小一致,因为其中只需要保存一个引用。而有些语言(例如c)采用的不是这种方式,它们把变量直接保存在变量的存储区里,这种方式就称为值语义。这样的话,一个整数类型的变量就需要保存一个整数所需要的空间(例如c语言中int类型占用4个字节大小)。

python中变量与对象的引用关系类似于c语言的指针变量与指向地址之间的关系。

在python的数据结构中,对象分为可变对象和不可变对象。基本数据类型如int、float,元祖tuple、str是不可变对象;list(列表)、dict(字典)、set(集合)是可变对象,可变对象存储的元素的引用其实是没有改变的,改变的是其引用指向的值。

采用引用语义存储的只是一个变量的值所在的内存地址,而不是这个变量的值本身。

1.2 第二点

现在说明第二点:Python中的变量、对象、引用三者之间的关系。

在Python里一切皆对象。Python中,对象具有三要素:标识(identity)、类型(type)、值(value)。

标识(identity):

用于唯一标识对象,通常对应对象在计算机内存中的地址。使用内置函数id(obj)返回对象唯一标识。

类型(type):

类型可以限制对象的取值范围和可执行的操作。使用内置函数type(obj)返回对象所属类型。

对象中含有标准的头部信息:类型标识符。标识对象类型,表示对象存储的数据的类型。

每一个对象都有两个标准的头部信息:

1.类型标识符,去标识对象的(数据)类型;

2.引用计数器,记录当前对象的引用的数目。

(回收机制:变量的引用计数器为0,自动清理。 ※ 较小整数型对象有缓存机制。)

值(value):

表示对象存储的数据的信息。使用内置函数print(obj)可以直接打印值。

Python中,变量用来指向任意的对象,是对象的引用。Python变量更像是指针(或者说Python变量更像“贴签”),而不是数据存储区域(而不是数据“容器”)。

Python 中的变量不是装有对象的“容器”,而是贴在对象上的“标签”——给一个变量赋值,把这个标签贴到一个对象上,重新赋值,是撕下标签贴到另一个对象上。

在python中,变量保存的是对象(值)的引用,采用这种方式,变量的每一次初始化,都开辟了一个新的空间,将新内容的地址赋值给变量。id()函数可以获取变量在内存中的地址。我们把不同的值赋给变量时候,地址发生变化,相同的值地址不发生变化。下面给出示例:

【顺便提示:id()的值不是固定不变的——此值系统为对象分配的内存地址,在你练习时显示的不同值是正常的。】

下面是字符串的示例:

在Python中,值可以放在内存的某个位置(地址),变量用于引用它们,给变量赋一个新值,原值不会被新值覆盖,变量只是引用了新值。顺便说明,Python的垃圾回收机制会自动清理不再被用到的值,所以不用担心计算机内存中充满被“丢弃”的无效的值。

1.3 第三点

现在说明第三点:可变(mutable) 类型对象、不可变(immutable) 类型对象

可变类型对象,指对象可以在其 id() 保持固定的情况下改变其取值。

不可变类型对象,指具有固定值的对象。不可变对象包括数字(numbers)、字符串(strings)和元组(tuples)。这样的对象不能被改变。如果必须存储一个不同的值,则必须创建新的对象。不可变对象不允许对自身内容进行修改。如果我们对一个不可变对象进行赋值,实际上是生成一个新对象,再让变量指向这个对象。哪怕这个对象简单到只是数字 0 和 1。

由于 Python 中的变量存放的是对象引用,所以对于不可变对象而言,尽管对象本身不可变,但变量的对象引用是可变的。运用这样的机制,有时候会让人产生糊涂,似乎可变对象变化了。如下面的代码:

i = 73 

i += 2

不可变的对象的特征没有变,依然是不可变对象,变的只是创建了新对象,改变了变量的对象引用。参见下图:

对于可变对象,其对象的内容是可以变化的。当对象的内容发生变化时,变量的对象引用是不会变化的。如下面的例子。

m=[5,9] 

m+=[6]

参见下图:

二、总结

Python变量指的是名字绑定了对象(绑定就是将一个对象与一个名字联系起来)。

绑定时,变量就是名字。

使用时,变量代表对象的引用。

变量改变的只有绑定关系。

深入学习:

https://docs.python.org/zh-cn/3.9/reference/datamodel.html#objects-values-and-types

补充说明:

对复杂的数据类型(列表、集合、字典),如果添加某一项元素,或者添加几个元素,不会改变其本身的地址,只会改变其内部元素的地址引用,但是如果对其重新赋值时,就会重新赋予地址覆盖就地址,这时地址就会发生改变。示例代码如下:

list_ = [1,2,3,4]
print(list_, id(list_))
list_.append(5)
print(list_, id(list_))
#如上代码,因为append前后的list_仍然是同一个对象,只是对象的值发了改变,所以地址不变。
 
#再如下面的代码
print(list_, id(list_), id(list_[1]))#打印列表、列表的地址、第二个元素的地址
list_[1] = 'aaa'   #修改列表
print(list_, id(list_), id(list_[1]))#打印列表、列表的地址、第二个元素的地址
#不难发发现:列表变了、列表的地址没有变、列表内部元素变了、列表内部元素的地址变了

测试运行如下图所示:

到此这篇Python变量的进阶分析就介绍到这了,更多Python学习内容请搜索W3Cschool以前的文章或继续浏览下面的相关文章。