scope python_Python标准库Scope

作者:zhbzz2007 出处:http://www.cnblogs.com/zhbzz2007 欢迎转载,也请保留这段声明。谢谢!

1 模块简介

你一定在很多计算机科学课程上听说过作用域。它很重要,如果你不理解它的工作原理,那么就会出现一些令人困惑的错误。作用域最基本的功能就是告诉编译器一个变量什么时候是可见的。也就是说,作用域定义了你使用变量的时间和范围。当你尝试使用一些不在当前作用域的变量时,你就会得到NameError。

Python有三类作用域:

局部作用域;

全局作用域;

非局部作用域(Python 3 中新增);

2 模块使用

2.1 局部作用域

局部作用域是Python中使用最多的作用域。当你在一段代码块中创建一个变量,它将会在最近的作用域中使用。所有的作用域组成的集合就是代码块环境。也就是说,默认是在局部作用域中处理所有的任务。如果你想要不同的作用域,那么你需要将变量设置为全局作用域或非局部作用域。

现在,我们使用Python的解释器创建一个简单的例子,来展示局部作用域任务。

>>> x = 10

>>> def my_func(a,b):

... print(x)

... print(z)

...

>>> my_func(1,2)

10

Traceback (most recent call last):

File "", line 1, in

File "", line 3, in my_func

NameError: global name 'z' is not defined

这里,我们创建了变量x以及一个入参为两个参数的简单函数。它将会打印x和z。请记住,我们还没有定义z,所以当我们调用这个函数时,我们将会获得NameError。这是因为z还没有定义或者它在作用域外部。如果你在调用函数之前定义z,那么就会发现z,你就不会获得NameError。

如果你尝试访问函数内部的变量,那么你也会获得NameError。

>>> def my_func(a,b):

... i = 2

... print(x)

...

>>> if __name__ == "__main__":

... x = 10

... my_func(1,2)

... print(i)

...

10

Traceback (most recent call last):

File "", line 4, in

NameError: name 'i' is not defined

变量i仅仅在函数内部定义,所以当你运行这段代码时,你将会得到NameError。

我们稍微修改一下上面的代码,将下面的代码存储到文件中,并运行。

def my_func(a,b):

x = 5

print(x)

if __name__ == "__main__":

x = 10

my_func(1,2)

print(x)

你认为将会发生什么?10打印两次?实际上并不是。原因就是我们现在有两个x变量。my_func函数中的变量x是局部函数作用域,它将会覆盖函数外部的变量x。当我们调用my_func函数时,我们打印5而非10。当函数返回时,my_func函数中的变量x会被回收,函数外的变量x将会起作用,这就是为什么最后一行语句打印出10。

如果你想了解具体的技巧,你可以在函数中的赋值语句前打印x,如下所示,

>>> def my_func(a,b):

... print(x)

... x = 5

... print(x)

...

>>> if __name__ == "__main__":

... x = 10

... my_func(1,2)

... print(x)

当你运行这段代码时,你将会得到如下的异常,

Traceback (most recent call last):

File "", line 3, in

File "", line 2, in my_func

UnboundLocalError: local variable 'x' referenced before assignment

这个异常发生,是因为Python提示你在my_func函数中后赋值给x,由于x还没有定义,因此它抛出异常。

2.2 全局作用域

Python包括global语句。它是Python的一个关键字。global语句定义了这个变量可以在随后的代码块中作为变量使用。虽然你可以在声明全局之前创建一个名称,但这是非常不鼓励的。让我们尝试使用global来修复上一个例子抛出的异常:

def my_func(a,b):

global x

print(x)

x = 5

print(x)

if __name__ == "__main__":

x = 10

my_func(1,2)

print(x)

这段代码的输出是,

10

5

5

通过将x定义为全局变量,我们告诉Pyton在函数中第一个打印函数中首先使用第一个定义的x。然后我们给x赋予新值5,在退出函数前再次打印。你将会注意到现在x是全局变量,当我们到达代码块的最后一个输出语句时,x依然是5。

让我们混合使用global和local来做一些有意思的事情,

def my_func(a,b):

global c

b,a = a,b

d = 'Mike'

print(a,b,c,d)

a , b , c , d = 1 , 2 , 'c is global' , 4

my_func(a,b)

print(a , b , c , d)

在这里,我们将变量c设置为全局变量。这个将会导致在函数内部和外部,c都会输出相同的值。我们在函数内部交换变量a和b,可以显示出我们在函数内部对其进行了交换,但是在函数外部,并没有修改二者。这也显示出变量a和b并不是全局变量,你应该可以看到如下的输出结果:

(2, 1, 'c is global', 'Mike')

(1, 2, 'c is global', 4)

我在此想提醒你不要在函数内部修改全局变量。这在Python社区中是一个不好的例子,它也会导致调试更加困难。

现在,我们已经理解了局部和全局变量,下面我们将要了解非局部变量。

2.3 非局部作用域

Python 3新增了一个关键词--nonlocal。关键词nonlocal增加了一个作用域用于覆盖内部作用域。你可以阅读PEP 3104。下面一段代码可以很好的解释非局部作用域。最常见的例子就是创建一个自增函数,

def counter():

num = 0

def incrementer():

num += 1

return num

return incrementer

如果你运行这段代码,你将会得到UnboundLocalError这个错误,因为变量num在内部函数中在赋值之前引用。让我们增加局部作用域,

>>> def counter():

... num = 0

... def incrementer():

... nonlocal num

... num += 1

... return num

... return incrementer

...

>>>

>>> c = counter()

>>> c

.incrementer at 0x7f67735ffea0>

>>> c()

1

>>> c()

2

>>> c()

3

现在我们定义的自增函数已按照我们期望开始工作。这种类型的函数被称为closure(闭包)。闭包就是一个将非局部变量封装起来的代码块。闭包背后的思想就是你可以在函数外部引用这些变量。

nonlocal允许你将变量分配到作用域外,但不是全局作用域。你不能在counter函数中使用nonlocal,因为它尝试着将其分配到全局作用域。你可以尝试一下,你将会得到SyntaxError。所以你必须在嵌套函数中使用nonlocal。

2.4 总结

在本文中,我们了解了通过Python关键词global和nonlocal来修改变量的引用方式。我们学习了在哪里使用以及为什么。我们也学习了局部作用域。

3 Reference

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/244440.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java命令_JAVA与模式之命令模式

在阎宏博士的《JAVA与模式》一书中开头是这样描述命令(Command)模式的:命令模式属于对象的行为模式。命令模式又称为行动(Action)模式或交易(Transaction)模式。命令模式把一个请求或者操作封装到一个对象中。命令模式允许系统使用不同的请求把客户端参数化&#xf…

android 16进制 全透明_你有几种实现方案Android 设备唯一标识?

前言项目开发中,多少会遇到这种需求:获得设备唯一标识DeviceId,用于:1.标识一个唯一的设备,做数据精准下发或者数据统计分析;2.账号与设备绑定;3.....分析这类文章,网上有许多资料&a…

链表的数据域怎么使用结构体_一步一步教你从零开始写C语言链表

为什么要学习链表?链表主要有以下几大特性:1、解决数组无法存储多种数据类型的问题。2、解决数组中,元素个数无法改变的限制(C99的变长数组,C也有变长数组可以实现)。3、数组移动元素的过程中,要对元素进行大范围的移动…

python计算bmi_Python编程语言:如何用Python编程来判断体重指数BMI是否健康

上一篇小编分享了自己学习Python语言有关字符串和模块time使用的相关知识,这一篇小编分享给大家的是比较有趣的运用,那就是如何用Python编程来表示自己体重BMI。 用Python程序来算出我们自己的BMI指数,来判断我们自己的健康情况,首…

drbd实现mysql地热备_heartheartbeat+drbd+mysql主库热备

1 环境主机名网卡磁盘mastereth0 桥接模式 eth0(192.168.1.10) 自定义模式(VMnet2)(192.168.2.10)VIP 192.168.1.200/210系统盘20G外接磁盘slaveeth0 桥接模式(192.168.1.20) eth1 自定义模式(VMnet2)(192.168.2.20)VIP 192.168.1.200/210系统盘20G外接磁盘server3eth0 桥接模式…

dba的前景_运维、测试、程序员,这些技术岗位哪个更有前景?

在一个初具规模的互联网公司,从业务方面出发,有很多岗位类型,比如运营、客服、市场、产品、设计、技术等等。在这些大类下面,还要细分各种小类,以技术为例,可分为前端(客户端)、后端、测试、运维、DBA等等&…

mysql深度解析_百万级数据下的mysql深度解析

mysql 作为一款非常优秀的免费数据库被广泛的使用,平时我们开发的项目数据过百万的时候不多。最近花了大量的时间来深入的研究mysql百万级数据情况下的优化。 遇到了很多问题并解决了他们,特此分享给大家。欢迎提出您的宝贵意见!一、百万级数…

python异步实现方式_Python通过yield实现异步

改写程序first函数等待long函数返回值的方式为yield,代码如下: import _thread import time gen None def long(): print (long execute) def fun(): time.sleep(5) result long end gen.send(result) _thread.start_new_thread(fun,()) def first(): print (firs…

免安装版的mysql步骤_mysql免安装版的安装方法及步骤

mysql免安装版的安装方法及步骤发布时间:2020-07-15 14:07:18来源:亿速云阅读:83作者:清晨小编给大家分享一下mysql免安装版的安装方法及步骤,希望大家阅读完这篇文章后大所收获,下面让我们一起去探讨吧&am…

jupyter notebook怎么写python代码_如何在Jupyter Notebook中使用Python虚拟环境?

如何在使用Jupyter Notebook时,解决Python虚拟环境间的切换问题?本文一步步帮你拆解。希望你能够避免踩坑的痛苦,把更多的时间花在愉快的编程上。 痛点 Python目前有两个主版本并存,这很让人苦恼。 一般人对于软件,总是…

cmd检查java_如何通过cmd查看java环境

展开全部JAVA环境变量设置一、下载JDK。62616964757a686964616fe4b893e5b19e31333363376561下载后是一个可执行程序,双击安装,安装路径为C:\Program Files\Java\jdk1.6.0_22\(当然,其他路径也可以),如下图:二、设置环境…

ios 检测是否联网_秋招|阿里 iOS 五轮面经分享,已收到阿里的意向书

作者:aaaaaazzzz链接:https://www.nowcoder.com/discuss/302113来源:牛客网感觉牛客很少看到iOS的面经了,今天收到了阿里的意向书,来分享下面经,希望大家都Offer~!总体感觉面试官都非常好&#…

java spring框架 注解_spring框架之注解的使用

原标题:spring框架之注解的使用今天是刘小爱自学Java的第122天。感谢你的观看,谢谢你。学习内容安排如下:Spring注解的使用。JavaWeb项目的搭建。Spring的Web集成。本来还计划学Spring的junit测试集成的,结果又没时间了。一、Spri…

idea 代码格式化插件_IDEA非常棒的插件,阿里巴巴约定成文的代码公约规范

无规矩,不方圆。每个人都有自己的编码风格,每个公司也有自己的代码规范。规范的代码,无论是自己日常维护,还是以后接盘者来接盘,都能快速定位上手,大大提高效率。作为一个IDEA万年爱好者,这些最…

java der pem_JAVA解析各种编码密钥对(DER、PEM、openssh公钥)

一、DER编码密钥对先说下DER编码,是因为JCE本身是支持DER编码密钥对的解析的,可以参见PKCS8EncodedKeySpec和X509EncodedKeySpec.DER编码是ASN.1编码规则中的一个子集,具体格式如何编排没有去了解,但最终呈现肯定的是一堆有规律的…

switch最大选项数目_随时随地学习C语言之3—if和switch哪个效率高?

之前学习C语言的时候,我经常有一个疑问,既然有if-else if-else结构的多分支选择语句,C语言为何还要制定switch这种多分支选择语句呢?直到两年前在分析ARM平台C语言反汇编代码的时候,才终于明白了switch-case这种结构存…

android java 退出程序_android开发两种退出程序方式(killProcess,System.exit)

KillProcess:在android中我们如果想要程序的进程结束可以这样写:android.os.Process.killProcess(android.os.Process.myPid());这样就可以从操作系统中结束掉当前程序的进程。注意:android中所有的activity都在主进程中,在Androi…

java 验证码_Java - 验证码 - 由Kaptcha组件实现

本文是基于SpringBoot整合Kaptcha验证码实现Kaptcha 是一个可高度配置的实用验证码生成工具,在项目开发中能够非常方便实现验证码先来看一个由 Kaptcha 制作的验证码效果图快速进入如何进行配置与实现的第1步:配置 Kaptcha 的依赖库com.github.penggle …

bat关闭java_通过bat脚本停止java进程的三个方法

通过bat脚本停止java进程的三个方法1.拷贝java.exe启动时复制java.exe程序修改进程映像名,然后通过映像名杀进程。栗子过程操作拷贝程序拷贝一份java.exe程序,重名为tstop.exe,放在PATH下。修改启动脚本echo offtitle tstoptstop -jar …/lib/laucher.ja…

onclick 获取img 里面的id_红魔5S游戏手机如何解锁bl获取第三方面具root权限

自从苹果阵营发布S系列,安卓系列也学习苹果的步伐,开始发布S系列,这不努比亚也发不了今年夏天的游戏手机,红魔5S游戏手机。红魔5S相对于红魔5其实从配置来看,三大件配置变化并不大,主要还是细节上的一些提升…