python 本地变量和全局变量 locals() globals() global nonlocal 闭包 以及和 scala 闭包的区别
最近看 scala ,看到了它的作用域,特此回顾一下python的变量作用域问题。
A = 10
B = 100
print A #10
print globals() #{'A': 10, 'B': 100, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', '__package__': None, '__name__': '__main__', '__doc__': None}
def mu(x,y):
B = 9
print locals() #{'y': 9, 'x': 10,'B':9}
if A == 10:
B = 19
print locals() #{'y':9,'x':10,'B':19}
return x+y
else:
return x-y
m = mu(10,9)
print(m) # 19
print locals() #{'A': 10, 'B': 100, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', 'm': 19, '__package__': None, 'mu': <function mu at 0x02ADA430>, '__name__': '__main__', '__doc__': None}
print globals()#{'A': 10, 'B': 100, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', 'm': 19, '__package__': None, 'mu': <function mu at 0x02ADA430>, '__name__': '__main__', '__doc__': None} # 全局作用域的 locals 和globals 内的变量内容是一样的。
print __name__ # __main__
print __file__ #E:/PycharmProjects/untitled/test1.py
print __doc__ # None
print __package__ # None
print B # 100
global
A = 10
B = 100
print A # 10
print globals()#{'A': 10, 'B': 100, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', '__package__': None, '__name__': '__main__', '__doc__': None}
def mu(x,y):
global B
B +=1
print(B) #101
print locals() #{'y': 9, 'x': 10} 进过global 作用, 本地变量中不会创建新的本地变量 B 了。而是直接用的全局的 B。
if A == 10:
B = 20
print locals() #{'y': 9, 'x': 10}
return x+y
else:
return x-y
m = mu(10,9)
print(m)
print locals() #{'A': 10, 'B': 20, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', 'm': 19, '__package__': None, 'mu': <function mu at 0x02A8A430>, '__name__': '__main__', '__doc__': None}
print globals()#{'A': 10, 'B': 20, '__builtins__': <module '__builtin__' (built-in)>, '__file__': 'E:/PycharmProjects/untitled/test1.py', 'm': 19, '__package__': None, 'mu': <function mu at 0x02A8A430>, '__name__': '__main__', '__doc__': None}
print B#20 因为函数中使用了 global 关键字, 函数中使用的是全局变量B ,并且对其做出了修改,所以这个B也就变成了 20
赋值 操作
A = 10
B = 100
print A
print globals()
def mu(x,y):
# B = B +1 # 这么做会报错 因为赋值操作会查找 本地变量 然而本地变量中并没有B 变量 并不是想象中的 B = 100 +1 ,只要出现赋值语句,就会遮蔽外层变量,而在此之前,本地变量中并未定义B。
# print(B) #100
print locals()
if A == 10:
if B == 100: #这么做不会报错,它会查找到全局变量
print "haha"
print locals() #{'y': 9, 'x': 10}
return x+y
else:
return x-y
m = mu(10,9)
print(m)
print locals()
print globals()
print B #100
以下为python3 操作
>>> def a(x):
... print(x)
... def b():
... print(x) # 参数作用域,父函数的参数可以 被子函数所使用,这也就是闭包。scala 作用域与此效果类似,内部函数,可以直接使用外部函数的变量。
... b()
...
>>>
>>> a(10)
10
10
>>> def a(x):
... print(x)
... def b():
... print(x) #这里会报错,因为在一旦在函数内部出现赋值操作,那么就会对外层的变量产生遮蔽效果,无论这赋值参数在本函数内部的位置是否在前,或者是在后,只要出现,就会出现遮蔽效果,此时如果在赋值操作之前使用该变量,就会出现未定义该变量的错误。 scala 的变量作用域效果与此类似。
... x = 20
... print(x)
... b()
...
>>> a(10)
10
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 7, in a
File "<stdin>", line 4, in b
UnboundLocalError: local variable 'x' referenced before assignment
>>> def a(x):
... print(x)
... def b():
... nonlocal x #python3 的 nonlocal 语句可以内部函数直接使用外部函数
... x = 20
... print(x)
... b()
... print(x)
...
>>> a(10)
10
20
20
>>> def a(x):
... print(locals())
... def b():
... print(x)
... print(locals())
... b()
...
>>> a(1)
{'x': 1}
1
{'x': 1}
>>> def a(x):
... print(locals())
... def b():
... print(locals()) #虽然赋值操作在本打印语句之后,但是仍然 可以看出,x 已经被遮蔽了
... x = 20 #由于未使用nonlocal,此时的x 是一个和外部x 同名的全新的本地变量,所以,对他的任何修改,仅限于本函数内部,不会影响到外部。
... print(locals())
... print(x)
... b()
... print(x) # 这是 遮蔽效果的证明
...
>>> a(10)
{'x': 10}
{}
{'x': 20}
20
10
>>>
闭包
def a():
x = 20
def b(y):
return y+x # 注意闭包内部是直接使用 外层函数的变量,而没有对外层函数变量进行赋值操作。
return b(10)
def c():
x = 20
def b(y):
x += 12 # 这里对x 进行了赋值操作。 x = x+12 ,这会报错,这其实已经不是闭包了,由于出现了赋值操作,所以,这里的 x 是一个和外层同名的变量,但是在本函数内部,本等式之前,并未进行 x 的宣告。所以会报错
return x +y
return b(10)
print(c())
贴一段scala 的闭包代码:
object PackageStudy {
def main(args: Array[String]): Unit = {
def a(): Int = {
var x = 10
def b(y: Int): Int = {
x += 13 // scala 可以做此操作,对捕获的自由变量进行了修改,但是 python 不行,因为这个操作在python 内部是一个重新宣告x,重新赋值的操作,这会产生屏蔽效果,python 宣告变量 是“贴签”操作。所以,python貌似是不支持对捕获变量的修改。也就没有 所谓在函数内部进行修改,会影响 外部的自由变量的值 这种情况的出现。
x + y
}
def c(z: Int): Int = {
x + z
}
println(x) //10
println(b(10)) //33 第一次捕获的自由变量的值是 x =10,所以此时的函数字面量中 x 绑定的是10
println(x) //23 //在闭包内部对捕获的自由变量的修改,也会影响到外面的自由变量。
println(c(3)) //26 自由变量经过 b() 的修改,c(3)再次捕获x时,x 已经是 23 了,此时的 函数字面量中绑定的x 是23
0
}
println(a())
}
}
python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 。
python 本地变量和全局变量 locals() globals() global nonlocal 闭包 以及和 scala 闭包的区别的更多相关文章
- Python 私有变量中两个下划线 _ _item 与 一个下划线的区别 _item
python中没有常量的说法, 但是可以通过元组实现一个常量 在python的私有变量中, 存在两个下划线 _ _item 与一个下划线 _item 的区别 前面带两个下划线的私有变量: 只能在本类中 ...
- Python 基础之返回值与函数使用与局部变量和全局变量locals() 和 globals()
一.函数的返回值 return return: 自定义返回值,返回到哪里? 返回到函数的[调用处]1.return 后面可以跟上六个标准数据类型,除此之外,可以跟上 类对象,函数,如果不写return ...
- 说几个python与c区别的地方以及静态变量,全局变量的区别
一: python代码: a = 2 def b(): print a a = 4 print a b() 在b函数中,有a=4这样的代码,说明a是函数b内部的局部变量,而不是外部的那个值为2的全局变 ...
- python中list作为全局变量无需global声明的原因
发现一个问题. python中list变量作为全局变量时,在函数中可以直接修改. 而普通变量则需要先在函数中global声明,否则会报错. 例如: a = 1 def fun(): global a ...
- PHP中全局变量的使用global和$GLOBALS[]
From: http://blog.csdn.net/happyqyt/article/details/7219889 用PHP开发项目,不可避免的会使用到全局变量,比如一些网站的配置信息,全站通用, ...
- python开发_python中的变量:全局变量和局部变量
如果你在为python中的变量:全局变量和局部变量头疼,我想这篇blog会给你帮助 运行效果: 代码部分: #Python中的变量:全局变量和局部变量 #在很多语言中,在声明全局变量的时候,都喜欢把全 ...
- python课堂整理10---局部变量与全局变量
一.局部变量与全局变量 1. 没有缩进,顶头写的变量为全局变量 2. 在子程序里定义的变量为局部变量 3. 只有函数能把变量私有化 name = 'lhf' #全局变量 def change_name ...
- C语言学习--全局变量、静态本地变量
全局变量 全局变量 定义在函数外面的变量是全局变量 全局变量具有全局的生存期和作用域 它们与任何函数无关 在任何函数内部都可以使用它们 #include <stdio.h> int f(v ...
- python 得到变量名的结果为名的变量的值locals()
>>> a="1">>> b="a">>> print(a,b)1 a>>> print ...
随机推荐
- struts中指定编码(使用Filter后仍然乱码)
https://www.cnblogs.com/oldinaction/p/5167481.html 概述: Tomcat默认是 ISO编码,不支持中文.尝试过自己写 Filter,在web.xml中 ...
- 查看端口 (windows)
查看端口 netstat -an |findstr
- 轻松理解execl系列函数
execl函数功能如下:启动一个可执行文件,并且对他进行传送参数.一些原型如下 #include <unistd.h> extern char **environ; int execl(c ...
- ubuntu 14.04 git clone 出现 fatal: Unable to find remote helper for 'https'
当你编译安装git时因为没有安装(lib)curl-devel所以导致git clone 和 git push 都会出现这个错误 如果你安装了(lib)curl-devel,然后重新编译安装git就没 ...
- spring Annotation based configuration
spring 注解相关 https://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch04s11.html
- bzoj2458 最小三角形
Description Xaviera现在遇到了一个有趣的问题.平面上有N个点,Xaviera想找出周长最小的三角形.由于点非常多,分布也非常乱,所以Xaviera想请你来解决这个问题.为了减小问题的 ...
- 渐变(Gradients)
渐变是一种可以在两个或两个以上颜色之间实现平稳过渡的效果,分为线性渐变(Linear Gradients)和径向渐变(Radial Gradients). 在演示之前,先创建一个div,并添加基础样式 ...
- 杂项:BugFree
ylbtech-杂项:BugFree BugFree是借鉴微软的研发流程和Bug管理理念,使用PHP+MySQL独立写出的一个Bug管理系统.简单实用.免费并且开放源代码(遵循GNU GPL). 命名 ...
- 学习笔记之Elasticsearch
Elasticsearch: RESTful, Distributed Search & Analytics | Elastic https://www.elastic.co/products ...
- 关于oracle数据库压力测试
今天接到需求,需要对oracle数据库进行压力测试,就这几个字的需求. 然后查看了以下软件: 1.Benchmark Factory是一款专业的服务器性能测试工具,专为数据库测试和可扩展性测量而设计, ...