跟老齐学Python之深入变量和引用对象


Posted in Python onSeptember 24, 2014

在《永远强大的函数》那一讲中,老齐我已经向看官们简述了一下变量,之后我们就一直在使用变量,每次使用变量,都要有一个操作,就是赋值。本讲再次提及这个两个事情,就是要让看官对变量和赋值有一个知其然和知其所以然的认识。当然,最后能不能达到此目的,主要看我是不是说的通俗易懂了。如果您没有明白,就说明我说的还不够好,可以联系我,我再为您效劳。

变量和对象

在《learning python》那本书里面,作者对变量、对象和引用的关系阐述的非常明了。我这里在很大程度上是受他的启发。感谢作者Mark Lutz先生的巨著。

应用《learning python》中的一个观点:变量无类型,对象有类型

在python中,如果要使用一个变量,不需要提前声明,只需要在用的时候,给这个变量赋值即可。这里特别强调,只要用一个变量,就要给这个变量赋值。

所以,像这样是不行的。

>>> x

Traceback (most recent call last):

  File "<stdin>", line 1, in <module>

NameError: name 'x' is not defined

 反复提醒:一定要注意看报错信息。如果光光地写一个变量,而没有赋值,那么python认为这个变量没有定义。赋值,不仅仅是给一个非空的值,也可以给一个空值,如下,都是允许的

>>> x = 3

>>> lst = []

>>> word = ""

>>> my_dict = {}

 在前面讲述中,我提出了一个类比,就是变量通过一根线,连着对象(具体就可能是一个int/list等),这个类比被很多人接受了,算是我老齐的首创呀。那么,如果要用一种严格的语言来描述,变量可以理解为一个系统表的元素,它拥有过指向对象的命名空间。太严肃了,不好理解,就理解我那个类比吧。变量就是存在系统中的一个东西,这个东西有一种能力,能够用一根线与某对象连接,它能够钓鱼。

对象呢?展开想象。在机器的内存中,系统分配一个空间,这里面就放着所谓的对象,有时候放数字,有时候放字符串。如果放数字,就是int类型,如果放字符串,就是str类型。

接下来的事情,就是前面说的变量用自己所拥有的能力,把对象和自己连接起来(指针连接对象空间),这就是引用。引用完成,就实现了赋值。

跟老齐学Python之深入变量和引用对象

看到上面的图了吧,从图中就比较鲜明的表示了变量和对象的关系。所以,严格地将,只有放在内存空间中的对象(也就是数据)才有类型,而变量是没有类型的。这么说如果还没有彻底明白,就再打一个比喻:变量就好比钓鱼的人,湖水里就好像内存,里面有好多鱼,有各种各样的鱼,它们就是对象。钓鱼的人(变量)的任务就是用某种方式(鱼儿引诱)把自己和鱼通过鱼线连接起来。那么,鱼是有类型的,有鲢鱼、鲫鱼、带鱼(带鱼也跑到湖水了了,难道是淡水带鱼?呵呵,就这么扯淡吧,别较真),钓鱼的人(变量)没有这种类型,他钓到不同类型的鱼。

这个比喻太烂了。凑合着理解吧。看官有好的比喻,别忘记分享。

同一个变量可以同时指向两个对象吗?绝对不能脚踩两只船。如果这样呢?

>>> x = 4

>>> x = 5

>>> x

5

 变量x先指向了对象4,然后指向对象5,当后者放生的时候,自动跟第一个对象4接触关系。再看x,引用的对象就是5了。那么4呢?一旦没有变量引用它了,它就变成了孤魂野鬼。python是很吝啬的,它绝对不允许在内存中存在孤魂野鬼。凡是这些东西都被看做垃圾,而对垃圾,python有一个自动的收回机制。

在网上找了一个图示说明,很好,引用过来(来源:http://www.linuxidc.com/Linux/2012-09/69523.htm)

>>> a = 100         #完成了变量a对内存空间中的对象100的引用

 如下图所示:

跟老齐学Python之深入变量和引用对象

然后,又操作了:

>>> a = "hello"

 如下图所示:

跟老齐学Python之深入变量和引用对象

原来内存中的那个100就做为垃圾被收集了。而且,这个收集过程是python自动完成的,不用我们操心。

那么,python是怎么进行垃圾收集的呢?在Quora上也有人问这个问题,我看那个回答很精彩,做个链接,有性趣的读一读吧。Python (programming language): How does garbage collection in Python work?

is和==的效果

以上过程的原理搞清楚了,下面就可以深入一步了。

>>> l1 = [1,2,3]

>>> l2 = l1

 这个操作中,l1和l2两个变量,引用的是一个对象,都是[1,2,3]。何以见得?如果通过l1来修改[1,2,3],l2引用对象也修改了,那么就证实这个观点了。

>>> l1[0] = 99      #把对象变为[99,2,3]

>>> l1              #变了

[99, 2, 3]

>>> l2             #真的变了??br />
[99, 2, 3]

 再换一个方式:

>>> l1 = [1,2,3]

>>> l2 = [1,2,3]

>>> l1[0] = 99

>>> l1

[99, 2, 3]

>>> l2

[1, 2, 3]

 l1和l2貌似指向了同样的一个对象[1,2,3],其实,在内存中,这是两块东西,互不相关。只是在内容上一样。就好像是水里长的一样的两条鱼,两个人都钓到了,当不是同一条。所以,当通过l1修改引用对象的后,l2没有变化。

进一步还能这么检验:

>>> l1

[1, 2, 3]

>>> l2

[1, 2, 3]

>>> l1 == l2    #两个相等,是指内容一样

True

>>> l1 is l2    #is 是比较两个引用对象在内存中的地址是不是一样

False          #前面的检验已经说明,这是两个东东
>>> l3 = l1
 #顺便看看如果这样,l3和l1应用同一个对象

>>> l3

[1, 2, 3]

>>> l3 == l1

True

>>> l3 is l1    #is的结果是True

True

 某些对象,有copy函数,通过这个函数得到的对象,是一个新的还是引用到同一个对象呢?看官也可以做一下类似上面的实验,就晓得了。比如:

>>> l1

[1, 2, 3]

>>> l2 = l1[:]

>>> l2

[1, 2, 3]

>>> l1[0] = 22

>>> l1

[22, 2, 3]

>>> l2

[1, 2, 3]
>>> adict = {"name":"qiwsir","web":"qiwsir.github.io"}

>>> bdict = adict.copy()

>>> bdict

{'web': 'qiwsir.github.io', 'name': 'qiwsir'}

>>> adict["email"] = "qiwsir@gmail.com"

>>> adict

{'web': 'qiwsir.github.io', 'name': 'qiwsir', 'email': 'qiwsir@gmail.com'}

>>> bdict

{'web': 'qiwsir.github.io', 'name': 'qiwsir'}

 不过,看官还有小心有点,python不总按照前面说的方式出牌,比如小数字的时候

>>> x = 2

>>> y = 2

>>> x is y

True

>>> x = 200000

>>> y = 200000

>>> x is y      #什么道理呀,小数字的时候,就用缓存中的.

False
>>> x = 'hello'

>>> y = 'hello'

>>> x is y

True

>>> x = "what is you name?"

>>> y = "what is you name?"

>>> x is y      #不光小的数字,短的字符串也是

False

 赋值是不是简单地就是等号呢?从上面得出来,=的作用就是让变量指针指向某个对象。不过,还可以再深入一些。走着瞧吧。

Python 相关文章推荐
python实现划词翻译
Apr 23 Python
python文件读写并使用mysql批量插入示例分享(python操作mysql)
Feb 17 Python
Python正则获取、过滤或者替换HTML标签的方法
Jan 28 Python
python发送邮件实例分享
Jul 28 Python
Python使用Selenium模块模拟浏览器抓取斗鱼直播间信息示例
Jul 18 Python
对numpy中的where方法嵌套使用详解
Oct 31 Python
django echarts饼图数据动态加载的实例
Aug 12 Python
python-tornado的接口用swagger进行包装的实例
Aug 29 Python
使用python制作一个解压缩软件
Nov 13 Python
Python数据分析pandas模块用法实例详解
Nov 20 Python
简单的Python人脸识别系统
Jul 14 Python
python 模块导入问题汇总
Feb 01 Python
Python greenlet实现原理和使用示例
Sep 24 #Python
跟老齐学Python之数据类型总结
Sep 24 #Python
跟老齐学Python之集合的关系
Sep 24 #Python
跟老齐学Python之集合(set)
Sep 24 #Python
跟老齐学Python之有点简约的元组
Sep 24 #Python
跟老齐学Python之dict()的操作方法
Sep 24 #Python
Python单链表的简单实现方法
Sep 23 #Python
You might like
Zerg基本策略
2020/03/14 星际争霸
PHP 5.0对象模型深度探索之类的静态成员
2008/03/27 PHP
基于PHP CURL获取邮箱地址的详解
2013/06/03 PHP
解决FastCGI 进程超过了配置的活动超时时限的问题
2013/07/03 PHP
php简单防盗链实现方法
2015/07/29 PHP
phpinfo的知识点总结
2019/10/10 PHP
用jquery写的一个万年历(自写)
2014/01/20 Javascript
js图片处理示例代码
2014/05/12 Javascript
jQuery自带的一些常用方法总结
2014/09/03 Javascript
js实现select下拉框菜单
2015/12/08 Javascript
实例讲解DataTables固定表格宽度(设置横向滚动条)
2017/07/11 Javascript
原生js获取left值和top值的三种方法
2017/08/02 Javascript
Vue项目中使用Vux的安装过程
2018/05/01 Javascript
mocha的时序规则讲解
2019/02/16 Javascript
vue 取出v-for循环中的index值实例
2019/11/09 Javascript
js实现时钟定时器
2020/03/26 Javascript
[01:22]DOTA2神秘商店携大量周边降临完美大师赛
2017/11/07 DOTA
[45:59]EG vs OG 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/18 DOTA
python基础教程之lambda表达式使用方法
2014/02/12 Python
Python守护进程(daemon)代码实例
2015/03/06 Python
Python使用Supervisor来管理进程的方法
2015/05/28 Python
简单总结Python中序列与字典的相同和不同之处
2016/01/19 Python
python访问抓取网页常用命令总结
2017/04/11 Python
Python numpy实现数组合并实例(vstack,hstack)
2018/01/09 Python
Django之提交表单与前后端交互的方法
2019/07/19 Python
python编写简单端口扫描器
2019/09/04 Python
Flask中endpoint的理解(小结)
2019/12/11 Python
Python 字符串池化的前提
2020/07/03 Python
深入浅析css3 border-image边框图像详解
2015/11/24 HTML / CSS
英国高级健康和美容产品零售商:Life and Looks
2019/08/01 全球购物
员工年终演讲稿
2014/01/03 职场文书
秋季运动会广播稿
2014/02/22 职场文书
医德医风演讲稿
2014/05/20 职场文书
经费申请报告
2015/05/15 职场文书
postgresql 删除重复数据案例详解
2021/08/02 PostgreSQL
Golang 并发下的问题定位及解决方案
2022/03/16 Golang