2024年最新一起刷完了这份1307页的Python面试宝典吧,python3的变量作用域规则和nonlocal关键字(1),2024年最新BAT面试文档

收集整理了一份《2024年最新Python全套学习资料》免费送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img



既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Python知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
img

正文

这段代码中函数中的i被绑定在局部作用域(也就是函数体内)中,在绑定的作用域中可见,并将外部的名字隐藏,而对一个未声明的局部变量赋值会导致错误,所以上面的代码会引发ReferenceError: i is not defined

对于python来说也是一样的问题,python代码在执行前首先会被编译成字节码,这就会导致某些时候实际执行的程序会和我们看到的产生出入。不过我们有dis模块帮忙,它可以输出python对象的字节码,下面我们就来看下经过编译后的f:

dis(f)

2 0 LOAD_GLOBAL 0 (print)

2 LOAD_FAST 0 (i)

4 CALL_FUNCTION 1

6 POP_TOP

3 8 LOAD_CONST 1 (‘a’)

10 STORE_FAST 0 (i)

4 12 LOAD_GLOBAL 0 (print)

14 LOAD_FAST 0 (i)

16 CALL_FUNCTION 1

18 POP_TOP

20 LOAD_CONST 0 (None)

22 RETURN_VALUE

其中LOAD_FAST和STORE_FAST是读取和存储local作用域的变量,我们可以看到,i变成了局部作用域的变量!而对i的赋值早于i的定义,所以报错了。

产生这种现象的原因也很简单,python对函数的代码是独立编译的,如果未加说明而在函数内对一个变量赋值,那么就认为你定义了一个局部变量,从而把外部的同名对象屏蔽了。这么做无可厚非,毕竟python没有独立的声明一个局部变量的语法,但结果就会造成我们看到的类似暂时性死区的现象。所以请允许我把es6的概念套用在python身上。

消除暂时性死区

既然知道问题的症结在于python无法区分局部变量的声明和定义,那么我们就来解决它。

对于一个可以区分声明和定义的语言来说是没有这种烦恼的,比如c:

‘’’

遇到问题没人解答?小编创建了一个Python学习交流QQ群:778463939

寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!

‘’’

int i = 0;

void f(void)

{

i++;

printf(“%d\n”, i); // 1

const char *i = “hello”;

printf(“%s\n”, i); // “hello”

}

python中不能这么做,但是我们可以换一个思路,声明一个变量是全局作用域的,这样不就解决了吗?

global运算符就是为了这个目的而存在的,它声明一个变量始终是全局作用域的变量,因此只要存在global声明,那么当前作用域里的这个名字就是一个对同名全局变量的引用。改进后的函数如下:

def f():

global i

print(i)

i += 1

print(i)

现在运行程序就会是你想要的结果了:

python test.py

0

1

1

如果你还是不放心,那么我们再来看看字节码:

dis(f)

3 0 LOAD_GLOBAL 0 (print)

2 LOAD_GLOBAL 1 (i)

4 CALL_FUNCTION 1

6 POP_TOP

4 8 LOAD_CONST 1 (‘a’)

10 STORE_GLOBAL 1 (i)

5 12 LOAD_GLOBAL 0 (print)

14 LOAD_GLOBAL 1 (i)

16 CALL_FUNCTION 1

18 POP_TOP

20 LOAD_CONST 0 (None)

22 RETURN_VALUE

对于i的存取已经由LOAD_GLOBALSTORE_GLOBAL接手了,没问题。

当然global也有它的局限性:

  • 一旦声明global,那么这个名字始终是global作用域的一个变量,不可以再是局部变量

  • 名字必须存在于global里,因为python在运行时进行名字查找,所以你的变量在global里找不到的话对它的引用将会出错

  • 接上一条,因为global限定了名字查找的范围,所以像闭包作用域的变量就找不到了

事实上需要引用非global名字的需求是极其常见的,因此为了解决global的不足,python3引入了nonlocal

使用nonlocal声明闭包作用域变量

假设我们有一个需求,一个函数需要知道自己被调用了多少次,最简单的实现就是使用闭包:

‘’’

遇到问题没人解答?小编创建了一个Python学习交流QQ群:778463939

寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!

‘’’

def closure():

count = 0

def func():

other code

count += 1

print(f’I have be called {count} times’)

return func

还是老问题,这样写对吗?

答案是不对,你又制造暂时性死区啦!

f=closure()

f()

Traceback (most recent call last):

File “”, line 1, in

File “”, line 5, in func

UnboundLocalError: local variable ‘count’ referenced before assignment

这时候就要nonlocal出场了,它声明一个名字位于闭包作用域中,如果闭包作用域中未找到就报错。

所以修正后的函数如下:

def closure():

count = 0

def func():

other code

nonlocal count

count += 1

print(f’I have be called {count} times’)

return func

测试一下:

一、Python所有方向的学习路线

Python所有方向路线就是把Python常用的技术点做整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照上面的知识点去找对应的学习资源,保证自己学得较为全面。

二、学习软件

工欲善其事必先利其器。学习Python常用的开发软件都在这里了,给大家节省了很多时间。

三、入门学习视频

我们在看视频学习的时候,不能光动眼动脑不动手,比较科学的学习方法是在理解之后运用它们,这时候练手项目就很适合了。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA56iL5bqP5aqb56eD56eD,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center)

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
[外链图片转存中…(img-gAqjDCHg-1713854721180)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 17
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值