python3中 exec 的局部变量获取失败问题解决过程记录

这几日,有粉丝问我一个报错问题,说他在某个calss内,进行exec的时候,频繁报错。经定位得知,exec的拼装字符串中,有个变量。但奇怪的是,这个变量并没有成功拼接进去。

我当即告诉他,这是因为exec函数中全局变量和局部变量的原因,直接获取局部变量是不行的,粉丝说他也在网上找了很久解决办法,也感觉是这个原因,但是一直没有解决,网上的写法都试了,都不行。

于是,我意识到这块的知识点,可能并没有很好的普及,于是写了本文来普及一下,请大家跟随我的文章,一点点探究弄懂这一块儿...

首先,我们先看这样一段代码:

运行之后,大家猜猜会输出什么?报错还是1? 答案:1

看到了吧?这就是exec的作用,可以执行某一段字符串。而且这个字符串中不但新建了变量b,还引用了变量a。


那我们现在来升级一下难度:

大家来猜测下输出是什么?报错还是5?

答案:报错...

看到了吧,这里就开始出现了局部变量的问题。


那我们现在假如给 这个b 在x函数中先声明一下,还会报错么?

输出答案:3

从这个结果我们可以看出,虽然没有报错,但exec的内容似乎并没有生效,b仍然是3。那我们是不是可以猜测,exec中声明的变量b,并非在这个x函数内成为局部变量,而是停留在最外层成为全局变量了呢?


于是,我们把这个print(b) 移动到了x函数外,看看输出:

答案:

是不是感到很意外?居然外侧的全局变量也没有?那这个exec中的b,似乎消失了?

原因是什么呢?请看以下解释:

exec() 是一个内置函数,可以动态用python来执行字符串,并返回表达式结果。

我们平时用法就是:exec(b=5) 这样简单的用法。

但,其实exec真正的官方写法是:

python 复制代码
exec(object[, globals[, locals]])

其中 globalslocals参数是可选的,用于指定全局变量和局部变量的命名空间。

也就是说,其实exec是可以指定变量的作用域的...


还是上面这个b的问题,我们现在给它改造一下(加个globals()),让其可以生成全局变量b:

打印结果:

可以看到已经变成了全局变量。

然后,我们引申一下,我们知道,当函数内向打印某个变量,但这个变量并非局部变量,而是全局变量的时候。函数内打印的就是全局变量的值:

结果:5

但是如果说,这个函数内已经有了一个局部变量b,那么输出的结果就是:

结果:3

此时,如果你在外面也打印b:

那么结果就是:3 5

到此,我们知道了如何利用exec生成全局变量。而使用全局变量可以随便用。


那问题来了,如何在exec中,使用全局变量或局部变量呢?

使用全局变量:

我们生成的变量b, 在exec中使用了全局变量a。

打印结果:1


那如果exec中使用的是局部变量a呢?这个问题就是粉丝一开始犯难的问题了,如果直接这么写:

那结果肯定是报错:

其实想要成功的办法很简单,而且有俩种写法:

写法一:让b成为全局变量后得到a的值。

结果:1

写法二:利用中间值tmp,来接收a的值,再从locals空间提取出来再赋值给b

结果:1


好,到此,关于exec在某个class或def中,到底应该怎么用。上述已经全部写明。

相关推荐
小鱼仙倌要会换位思考呀5 分钟前
焦距的作用是什么,为什么要引入焦距?
后端
Hello kele8 分钟前
大型项目,选择conda还是Poetry要点分析
人工智能·python·conda·ai编程·poetry
用户4192559999629 分钟前
mk-C/C++气象数据中心实战,手把手教你做工业级项目(完结)
后端
WeiLai111210 分钟前
面试基础---Redis 延迟队列深度解析
java·redis·分布式·后端·junit·面试·架构
SmallBambooCode11 分钟前
【人工智能】【Python】在Scikit-Learn中使用KNN(K最近邻算法)
人工智能·python·机器学习·scikit-learn·近邻算法
jaffe—fly14 分钟前
【解决问题】conda 虚拟环境内,`pip list` 展示全局的包
python·conda·pip
带上一无所知的我14 分钟前
解锁Conda:Python环境与包管理的终极指南
开发语言·python·conda
asphyxia16 分钟前
ajax前后端分离品牌管理项目
后端
用户861318977427917 分钟前
第8章 系统质量属性与架构评估篇(二)——软件系统质量属性
后端
程序员蜗牛18 分钟前
简化Java多线程编程的实用利器!
后端