•  当函数内部引用一个全局变量时,如果此全局变量为可变类型,那么函数内部是可以改变此全局变量的值,用不用globale申明全局变量都一样。但是如果想给此变量重新赋值则必须要使用global。
l = []
d = {}
def add_value():
d['frank'] = 'yangcaho'
l.append(d)
add_value()
print(l)
print(d) 输出:
[{'frank': 'yangcaho'}]
{'frank': 'yangcaho'}
l = []
d = {}
def add_value():
global l
global d
d['frank'] = 'yangcaho'
l.append(d)
add_value()
print(l)
print(d) 输出:
[{'frank': 'yangcaho'}]
{'frank': 'yangcaho'}
  •  当函数内部引用一个全局变量时,如果此全局变量为不可变类型,那么如果要能够在函数内部改变全局变量的值,就需要用global,不用改变则不需要用global也可以引用
a = 'frank'
def add_value():
global a
a = 'yangchao'
add_value()
print(a) yangchao
a = 'frank'
def add_value():
b = a + 'yangchao'
print(b)
add_value()
print(a) frankyangchao
frank

  结论:可以使用定义在函数外的全局变量的值(假设在函数内没有同名的变量)。但强烈建议尽量避免这样做,因为这使得程序的读者会不清楚这个变量是在哪里定义的。使用global语句可以清楚地表明变量是在外面的块定义的。

一些细微区别

可变对象如list的.paaend()操作与‘+’操作,的一些不同点。

  • 用 ‘+’ 实际上是重新定义一个变量了。但是原列表的内存位置不会变,会开辟一个新的内存,L不是直接在原来l的基础上增加然后指向l
l = ['frank', 'liuilu']
print(id(l))
L = l + ['yangchao']
print(id(L))
print(l)
print(id(l)) 36364616
36365960
['frank', 'liuilu']
36364616
  • 用 append 和 ’ += ‘实际上是在原来变量基础上增加。
l = ['frank', 'liuilu']
print(id(l))
l += ['yangchao']
print(l)
print(id(l)) 43639112
['frank', 'liuilu', 'yangchao']
43639112
  •  一下代码中个很坑的地方:为什么l的值不是[{'age': 1}, {'age': 2}]而是[{'age': 2}, {'age': 2}],因为l中存入的值是d实际是存入了d所指向内存地址。在第二次改变d的值得时候,就是改变了d的指向。于是第一次添加的d的值也会改变。
l = []
d = {}
def add_value():
for i in range(1, 3):
print('第%s次循环' %i)
d['age'] = i
print(d)
print('#' * 10)
print(l)
print('#' * 10)
l.append(d)
print(l)
add_value() 输出结果
第1次循环
{'age': 1}
##########
[]
##########
[{'age': 1}]
第2次循环
{'age': 2}
##########
[{'age': 2}]
##########
[{'age': 2}, {'age': 2}]

  但是函数内如果是用赋值的方式(如下面两种模式),那么函数中的d就不再是全局变量中的d了。所以有了每次赋值操作(即’=‘),每次append的d都是新的值。不会改变原来的值。

l = []
d = {}
def add_value():
for i in range(1, 3):
print('第%s次循环' %i)
d = {'age', i}
print(d)
print('#' * 10)
print(l)
print('#' * 10)
l.append(d)
print(l)
add_value() 第1次循环
{1, 'age'}
##########
[]
##########
[{1, 'age'}]
第2次循环
{2, 'age'}
##########
[{1, 'age'}]
##########
[{1, 'age'}, {2, 'age'}]
l = []
def add_value():
for i in range(1, 3):
d = {}
print('第%s次循环' %i)
d['age'] = i
print(d)
print('#' * 10)
print(l)
print('#' * 10)
l.append(d)
print(l)
add_value() 第1次循环
{'age': 1}
##########
[]
##########
[{'age': 1}]
第2次循环
{'age': 2}
##########
[{'age': 1}]
##########
[{'age': 1}, {'age': 2}]

  总得来讲就是在操作可变对象的时候,只要有 = 就会重新开辟内存创建新变量,诸如list.append() 、list += 、list.pop、dic.update()、dict['key'] = value、实际上都是在原来变量的基础上作修改,如果后面再次引用则会引用每次修改后的值。

  

a = []
b = []
c = {}
d = {}
e = 'frank'
f = 'frank'
print(id(a), id(b), id(c), id(d), id(e), id(f)) 43770760 43770824 31284896 31284968 31302408 31302408

  可见每次创建一个空的可变对象时,都会重新开辟内存,但不可变对象就是指向原来的的地址空间。

python中可变与不可变类型变量中一些较难理解的地方的更多相关文章

  1. python中的可变与不可变对象

    Python中的可变对象和不可变对象 什么是可变/不可变对象 不可变对象,该对象所指向的内存中的值不能被改变.当改变某个变量时候,由于其所指的值不能被改变,相当于把原来的值复制一份后再改变,这会开辟一 ...

  2. Python中的可变、不可变对象和赋值技巧序列解包

    可变对象和不可变对象 在python中一切皆对象.在Python中不存在所谓的值传递调用,一切传递都是对象的引用,也可认为是传址. python中,对象分为可变(mutable)和不可变(immuta ...

  3. python中不可变数据类型和可变数据类型

    在学习python过程中我们一定会遇到不可变数据类型和可变数据类型. 1.名词解释 以下所有的内容都是基于内存地址来说的. 不可变数据类型: 当该数据类型的对应变量的值发生了改变,那么它对应的内存地址 ...

  4. Python中可变和不可变类型

    可变类型 列表,字典,集合 不可变类型 数字,字符串,元组 这里的可变不可变,是指内存中的那块内容(value)是否可以被改变 不可变类型 数字 a = 1 b = 1 print(id(a), id ...

  5. php学习笔记5--php中的可变变量,可变函数及匿名函数

    可变变量指的是:将一个变量的值再次当做一个变量名从而得到另外一个变量的值.如:$name = 'dqrcsc';$myname = 'name'; //$myname的值碰巧是另一个变量的变量名ech ...

  6. iOS中怎么判断可变和不可变的坑(更正版)

    iOS中怎么判断可变和不可变的坑 怎么判断NSString和NSMutableString呢 看题 BOOL result = [" isKindOfClass:[NSMutableStri ...

  7. 【翻译】为什么Java中的String不可变

    笔主前言: 众所周知,String是Java的JDK中最重要的基础类之一,在笔主心中的地位已经等同于int.boolean等基础数据类型,是超越了一般Object引用类型的高端大气上档次的存在. 但是 ...

  8. C中參数个数可变的函数

    一.什么是可变參数 我们在C语言编程中有时会遇到一些參数个数可变的函数,比如printf()函数,其函数原型为: int printf( const char* format, ...); 它除了有一 ...

  9. java中的final与可变类型、不可变类型的关系

    如果你对final和不可变类型的概念与区别有疑问的话,可以打开这篇文章.希望我的解答可以帮到您! 1.不可变类型: 什么是可变类型,什么是不可变类型呢? 首先我们看一下下面的这行代码: String ...

随机推荐

  1. spring-boot-starter-mail技术总结

    1. spring-boot-starter-mail技术总结 1.1. 配置读取类SMTPTransport 在application中需要配置的信息,在此类中都可以看到,可以在此类打断点查看 1. ...

  2. 【机器学习】--xgboost从初识到应用

    一.前述 在 Kaggle 的很多比赛中,我们可以看到很多 winner 喜欢用 xgboost,而且获得非常好的表现,今天就来看看 xgboost 到底是什么以及如何应用.Gradient boos ...

  3. 【机器学习】--Adaboost从初始到应用

    一.前述 AdaBoost算法和GBDT(Gradient Boost Decision Tree,梯度提升决策树)算法是基于Boosting思想的机器学习算法.在Boosting思想中是通过对样本进 ...

  4. [个人网站搭建]·Django增加评论功能(Python3)

    [个人网站搭建]·Django增加评论功能 个人主页--> https://xiaosongshine.github.io/ 个人网站搭建github地址:https://github.com/ ...

  5. count()聚合函数正确用法

    count()聚合计算 count()是聚合函数,对于返回的结果集,一行行地判断,累计值加1,最后返回累计值,count(*).count(主键ID)和count(1)表示返回满足条件的结果集的总行数 ...

  6. 痞子衡嵌入式:极易上手的可视化wxPython GUI构建工具(wxFormBuilder)

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是wxPython GUI构建工具wxFormBuilder. 一.手工代码布局GUI界面的烦恼 如果你曾经设计过上位机软件GUI界面,初 ...

  7. .net core api +swagger(一个简单的入门demo 使用codefirst+mysql)

    前言: 自从.net core问世之后,就一直想了解.但是由于比较懒惰只是断断续续了解一点.近段时间工作不是太忙碌,所以偷闲写下自己学习过程.慢慢了解.net core 等这些基础方面学会之后再用.n ...

  8. ajax的嵌套需要注意的问题

    当我们要嵌套ajax的时候,需要注意 异步/同步 的处理,一般是要设置成同步,如果是异步,那么被嵌套的ajax的操作很可能获取不到想要的值,因为他可能比嵌套他的ajax跑的更早 在ajax中有一个as ...

  9. Java中nextLine()与其他next(),nextInt(),nextFloat()的区别

    最近学习Java过程中遇到一个小问题,就是用nextInt()来接收输入内容时,按完回车之后会产生后面的内容无法输入的结果,因此来做个小记录.有不足的地方还请大家指出. 区别一  读取返回结果 nex ...

  10. Linux运维第二课----Linux发展史、环境准备

    一.Linux的发展 1.1969年在贝尔实验室诞生Unix,是开源免费的,之后逐渐转变为收费系统. 2.1986年谭邦宁研发mini Unix,但主要用来教学. 3.斯托曼创建FSF(自由软件基金会 ...