python里面的全局变量和局部变量的区别(很好的一篇文章,找不到出处,原作者看到请联系,我会署上名字)
2015-03-20 19:54
651 查看
python中,对于变量作用域的规定有些不一样。在诸如C/C++、java等编程语言中,默认在函数的内部是可以直接访问在函数外定义的全局变量的,但是这一点在python中就会有问题,下面是一个例子。
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
COUNT = COUNT + 1
func()
Python test.py,会运行报错:
Traceback (most recent call last):
File "test.py", line 8, in <module>
func()
File "test.py", line 6, in func
COUNT = COUNT + 1
UnboundLocalError: local variable 'COUNT' referenced before assignment
“UnboundLocalError: local variable 'COUNT' referenced before assignment”的意思是变量COUNT在赋值之前被引用。
这里要知道python和其它编程语言不一样的地方。像C/C++之类的编程语言,变量名称实际上是代表的一块内存区域,对该变量赋值的意思就是将新的值放入该变量指定的内存区域。而对于python来说,所有的变量都是对内存区域的引用,对变量赋值相当于将变量引用的内存从一块区域改变到另外一块存放新值的区域。也就是说,C/C++中,变量名和内存区域的对应关系不会变,变的只是对应内存中存放的值;而在python中,变量只是对存放其值的内存区域的引用,变量值的改变不是因为变量指向的内存区域中的值发生了变化,而是变量引用了新的存放新值的内存区域。python中的所有变量都是相当于java中的不可变的变量,任何一次值的改变都对应着变量引用内存区域的变化。区别如下图1:
图1 变量的比较
python中有一个id函数,python中有一个id函数,help(id)可以看到它的说明,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
Help on built-in function id in module __builtin__:
id(...)
id(object) -> integer
Return the identity of an object. This is guaranteed to be unique among simultaneously existing objects.(Hint: it's the object's memory address.)
(END)
简单地说,id函数反应的是对象的内存地址,看下面的实验结果:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT = 1
for i in range(5):
COUNT = COUNT + 1
print id(COUNT)
python test.py运行结果:
11031328
11031304
11031280
11031256
11031232
这里和上面图上说明的相吻合,python中每一次赋值都使变量引用的内存空间发生了改变。
回到上面“referenced before assignment”的错误,之所以会发生这种错误是因为python在函数中发现对于COUNT变量的赋值,会将其添加到函数的局部命名空间(实际上,这是在函数运行到赋值操作之前发生的)。进行赋值操作时,赋值操作符的右边引用了COUNT变量,而这时COUNT变量只是被添加到了函数的局部命名空间,而没有被具体赋值,所以会发生上面的错误。实际上,这里问题就出在赋值操作的地方,因为有赋值操作导致该变量被添加到了函数的局部命名空间。如果没有赋值,只是引用该变量,是没有什么问题的,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
temp = COUNT + 1
print "temp:",COUNT
print "COUNT:",COUNT
func()
python test.py运行结果:
temp: 1
COUNT: 1
这样,COUNT变量没有被添加到函数的局部命名空间,python解释器在函数的局部命名空间中没有查找到它,然后,python解释器会继续在全局的命名空间中查找,结果在全局命名空间中找到COUNT的定义并引用它的值,所以程序运行没有任何问题。
到这里你可能会问,难道在函数中没法修改全局变量的值吗?不是的,如果要在函数中修改全局变量的值,就要在函数中对该变量进行global声明,以告诉python解释器,该变量是全局命名空间中的,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
global COUNT
COUNT = COUNT + 1
print "COUNT:",COUNT
func()
python test.py运行结果:
COUNT: 2
OK. ^_^
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
COUNT = COUNT + 1
func()
Python test.py,会运行报错:
Traceback (most recent call last):
File "test.py", line 8, in <module>
func()
File "test.py", line 6, in func
COUNT = COUNT + 1
UnboundLocalError: local variable 'COUNT' referenced before assignment
“UnboundLocalError: local variable 'COUNT' referenced before assignment”的意思是变量COUNT在赋值之前被引用。
这里要知道python和其它编程语言不一样的地方。像C/C++之类的编程语言,变量名称实际上是代表的一块内存区域,对该变量赋值的意思就是将新的值放入该变量指定的内存区域。而对于python来说,所有的变量都是对内存区域的引用,对变量赋值相当于将变量引用的内存从一块区域改变到另外一块存放新值的区域。也就是说,C/C++中,变量名和内存区域的对应关系不会变,变的只是对应内存中存放的值;而在python中,变量只是对存放其值的内存区域的引用,变量值的改变不是因为变量指向的内存区域中的值发生了变化,而是变量引用了新的存放新值的内存区域。python中的所有变量都是相当于java中的不可变的变量,任何一次值的改变都对应着变量引用内存区域的变化。区别如下图1:
图1 变量的比较
python中有一个id函数,python中有一个id函数,help(id)可以看到它的说明,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
Help on built-in function id in module __builtin__:
id(...)
id(object) -> integer
Return the identity of an object. This is guaranteed to be unique among simultaneously existing objects.(Hint: it's the object's memory address.)
(END)
简单地说,id函数反应的是对象的内存地址,看下面的实验结果:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT = 1
for i in range(5):
COUNT = COUNT + 1
print id(COUNT)
python test.py运行结果:
11031328
11031304
11031280
11031256
11031232
这里和上面图上说明的相吻合,python中每一次赋值都使变量引用的内存空间发生了改变。
回到上面“referenced before assignment”的错误,之所以会发生这种错误是因为python在函数中发现对于COUNT变量的赋值,会将其添加到函数的局部命名空间(实际上,这是在函数运行到赋值操作之前发生的)。进行赋值操作时,赋值操作符的右边引用了COUNT变量,而这时COUNT变量只是被添加到了函数的局部命名空间,而没有被具体赋值,所以会发生上面的错误。实际上,这里问题就出在赋值操作的地方,因为有赋值操作导致该变量被添加到了函数的局部命名空间。如果没有赋值,只是引用该变量,是没有什么问题的,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
temp = COUNT + 1
print "temp:",COUNT
print "COUNT:",COUNT
func()
python test.py运行结果:
temp: 1
COUNT: 1
这样,COUNT变量没有被添加到函数的局部命名空间,python解释器在函数的局部命名空间中没有查找到它,然后,python解释器会继续在全局的命名空间中查找,结果在全局命名空间中找到COUNT的定义并引用它的值,所以程序运行没有任何问题。
到这里你可能会问,难道在函数中没法修改全局变量的值吗?不是的,如果要在函数中修改全局变量的值,就要在函数中对该变量进行global声明,以告诉python解释器,该变量是全局命名空间中的,如下:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
test.py:
#!/usr/bin/python
COUNT=1
def func():
global COUNT
COUNT = COUNT + 1
print "COUNT:",COUNT
func()
python test.py运行结果:
COUNT: 2
OK. ^_^
相关文章推荐
- python里面的全局变量和局部变量的区别小讲
- C++ 局部静态变量,全局变量,全局静态变量,局部变量的区别和联系
- python的局部变量和全局变量区别
- Python全局变量与局部变量的区别
- 全局变量和局部变量在内存里的区别?堆和栈(转载-无法确定作者与出处) 收藏
- 关于工作与生活——别人多次转载的文章,找不到原来的出处了。但觉得文章很好,就转载了。对原作者表示歉意和敬意。
- python全局变量-局部变量用法和区别
- Python全局变量与局部变量区别及用法分析
- python全局变量-局部变量区别及用法
- 关于全局变量、静态全局变量、静态局部变量的区别与联系
- 一篇告诫正在学习以及将要学习计算机的同学的文章【找不到原作者及出处非常抱歉】
- C++ 局部静态变量,全局变量,全局静态变量,局部变量的区别和联系
- 不记得在哪看到的一篇文章。蛮好的,如果作者看到了联系我,我好写明引用地址,原作者
- python之全局变量和局部变量的区别
- 全局变量,静态全局变量,局部变量和静态局部变量的区别和联系
- Python入门的36个例子——14 局部变量与全局变量
- 全局变量、静态全局变量、静态局部变量和局部变量的区别
- 全局变量、局部变量、静态全局变量、静态局部变量的区别
- 2010.4.8 全局的静态的变量和局部变量的区别
- static全局变量,局部变量,函数与普通全局变量,局部变量和函数的区别