PYTHON exec() 函数 变量作用域问题浅析总结

2024-05-25 08:36

本文主要是介绍PYTHON exec() 函数 变量作用域问题浅析总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. exec(‘拼接字符串’,globals, locals)函数作用

exec() 可在python 中通过传入字符串的方式,从而执行字符串内的各种命令或表达式

---eval() 函数 与exec() 基本功能相同,唯一的区别,eval() 只可用于表达式计算并返回生成的结果,所以eval() 不可以执行没有结果的命令或表达式 如 ‘x= 1+2’不可在eval中执行

但exec() 所有操作都在内部执行,不返回任何结果。

2. exec() 在主线程与函数内的变量作用域问题

以定义一个变量x为例 :x= 3

a.在主线成中 x=3 与 exec('x=3')的结果是相同的,相当于定义了一个全局变量, x在程序内任意地点都可用

b. 在函数中定义x =3 与  exec('x=3')结果却不同,x=3表示在函数内定义了一个局部变量,而 exec('x=3')表示在exec() 自己特殊的执行环境中定义了一个局部变量,当exec()执行完成后,局部变量不可被外层函数再次访问,除非再次执行exec()函数,进行调用。 

****问题点1***** exec()执行完后会将定义的局部变量写入外层函数的locals() 局部变量字典表中,但直接查询(如使用print()) 无法获得对应值, 只能通过locals()['变量名']的方式获取

****原因***** print()等可以调用查找变量的方式是先查询当前函数在编译的时候已确定的变量名是否在内存栈中,再查询对应局部变量表中的具体值。exec() 在编译的时候,传入的命令是字符串形式的,无法被解析成变量,所以无法在预编译的内存中被找到,所以即使locals()字典中有对应变量,也会提示无法找到。

****注意1***** 函数中如果包含局部变量定义,则exec中如果定义同名的变量,那exec中指定的局部变量也无法重新将locals的值更新。 即使局部变量声明位置是在exec()位置之后,exec()函数也无法提前写入locals字典中

--------------------------------------------------

****问题点2***** 既然exec() 在函数中无法更新变量字典,那为什么在主线或全局变量中,声明就可以直接访问

****原因***** 因为局部变量的是每次调用到函数时将变量在内存栈空间创建位置,而 全局变量直接是在堆空间开辟位置,不会因为exec()结束而消失,所以可以提前调用

****深层原因点***** exec()函数中有两个可选参数,globals和locals 分别是指定 exec()内执行的命令中指定的全局变量和局部变量存储的位置,且必须为字典结构即 键-值对。

a. 当不指定 globals和locals两个参数时,exec() 默认使用当前空间(主线或者函数)中的globals()字典和locals()字典

b.  当指定 globals 但不指定locals参数时,exec() 默认使用指定的globals参数空间作为locals参数,而不是当前环境的locals()字典  

c.  当分别指定 globals 和指定locals两参数时,exec() 使用各自变量空间字典,互不干扰

理论上讲,exec()内创建的变量,函数等都是存入locals 指定的字典中。

但在主线函数中,全局变量空间与局部变量空间指向的内存空间相同,或者说 局部变量空间是全局变量空间的引用复制。 所以exec()存入locals中也就是存入了全局变量空间。

但在函数中,当不指定两个参数时,两个字典指向的不同变量字典,所以无法实现在主线函数中的声明变量效果。所以通过只指定一个globals()空间,即可解决这个问题。

结论

正式环境中,exec()的执行方式存在很大的安全隐患,所以可以理解为pyhthon中的一种bug,在生产环境中尽量减少使用。

参考文章

【实测】一文说懂:python3中exec 的局部变量获取失败问题处理办法【实测】一文说懂:python3中exec 的局部变量获取失败问题处理办法-腾讯云开发者社区-腾讯云

Python进阶:你定义的变量到底保存在哪里Python进阶:你定义的变量到底保存在哪里-腾讯云开发者社区-腾讯云

Python与家国天下

Python与家国天下-腾讯云开发者社区-腾讯云

这篇关于PYTHON exec() 函数 变量作用域问题浅析总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1001000

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

C# 中变量未赋值能用吗,各种类型的初始值是什么

对于一个局部变量,如果未赋值,是不能使用的 对于属性,未赋值,也能使用有系统默认值,默认值如下: 对于 int 类型,默认值是 0;对于 int? 类型,默认值是 null;对于 bool 类型,默认值是 false;对于 bool? 类型,默认值是 null;对于 string 类型,默认值是 null;对于 string? 类型,哈哈,没有这种写法,会出错;对于 DateTime 类型,默

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

2024.6.24 IDEA中文乱码问题(服务器 控制台 TOMcat)实测已解决

1.问题产生原因: 1.文件编码不一致:如果文件的编码方式与IDEA设置的编码方式不一致,就会产生乱码。确保文件和IDEA使用相同的编码,通常是UTF-8。2.IDEA设置问题:检查IDEA的全局编码设置和项目编码设置是否正确。3.终端或控制台编码问题:如果你在终端或控制台看到乱码,可能是终端的编码设置问题。确保终端使用的是支持你的文件的编码方式。 2.解决方案: 1.File -> S

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

十五.各设计模式总结与对比

1.各设计模式总结与对比 1.1.课程目标 1、 简要分析GoF 23种设计模式和设计原则,做整体认知。 2、 剖析Spirng的编程思想,启发思维,为之后深入学习Spring做铺垫。 3、 了解各设计模式之间的关联,解决设计模式混淆的问题。 1.2.内容定位 1、 掌握设计模式的"道" ,而不只是"术" 2、 道可道非常道,滴水石穿非一日之功,做好长期修炼的准备。 3、 不要为了

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

Python 字符串占位

在Python中,可以使用字符串的格式化方法来实现字符串的占位。常见的方法有百分号操作符 % 以及 str.format() 方法 百分号操作符 % name = "张三"age = 20message = "我叫%s,今年%d岁。" % (name, age)print(message) # 我叫张三,今年20岁。 str.format() 方法 name = "张三"age